1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * HID driver for gaming keys on Logitech gaming keyboards (such as the G15)
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (c) 2019 Hans de Goede <hdegoede@redhat.com>
6*4882a593Smuzhiyun */
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun #include <linux/device.h>
9*4882a593Smuzhiyun #include <linux/hid.h>
10*4882a593Smuzhiyun #include <linux/module.h>
11*4882a593Smuzhiyun #include <linux/random.h>
12*4882a593Smuzhiyun #include <linux/sched.h>
13*4882a593Smuzhiyun #include <linux/usb.h>
14*4882a593Smuzhiyun #include <linux/wait.h>
15*4882a593Smuzhiyun
16*4882a593Smuzhiyun #include "hid-ids.h"
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #define LG_G15_TRANSFER_BUF_SIZE 20
19*4882a593Smuzhiyun
20*4882a593Smuzhiyun #define LG_G15_FEATURE_REPORT 0x02
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #define LG_G510_FEATURE_M_KEYS_LEDS 0x04
23*4882a593Smuzhiyun #define LG_G510_FEATURE_BACKLIGHT_RGB 0x05
24*4882a593Smuzhiyun #define LG_G510_FEATURE_POWER_ON_RGB 0x06
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun enum lg_g15_model {
27*4882a593Smuzhiyun LG_G15,
28*4882a593Smuzhiyun LG_G15_V2,
29*4882a593Smuzhiyun LG_G510,
30*4882a593Smuzhiyun LG_G510_USB_AUDIO,
31*4882a593Smuzhiyun };
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun enum lg_g15_led_type {
34*4882a593Smuzhiyun LG_G15_KBD_BRIGHTNESS,
35*4882a593Smuzhiyun LG_G15_LCD_BRIGHTNESS,
36*4882a593Smuzhiyun LG_G15_BRIGHTNESS_MAX,
37*4882a593Smuzhiyun LG_G15_MACRO_PRESET1 = 2,
38*4882a593Smuzhiyun LG_G15_MACRO_PRESET2,
39*4882a593Smuzhiyun LG_G15_MACRO_PRESET3,
40*4882a593Smuzhiyun LG_G15_MACRO_RECORD,
41*4882a593Smuzhiyun LG_G15_LED_MAX
42*4882a593Smuzhiyun };
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun struct lg_g15_led {
45*4882a593Smuzhiyun struct led_classdev cdev;
46*4882a593Smuzhiyun enum led_brightness brightness;
47*4882a593Smuzhiyun enum lg_g15_led_type led;
48*4882a593Smuzhiyun u8 red, green, blue;
49*4882a593Smuzhiyun };
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun struct lg_g15_data {
52*4882a593Smuzhiyun /* Must be first for proper dma alignment */
53*4882a593Smuzhiyun u8 transfer_buf[LG_G15_TRANSFER_BUF_SIZE];
54*4882a593Smuzhiyun /* Protects the transfer_buf and led brightness */
55*4882a593Smuzhiyun struct mutex mutex;
56*4882a593Smuzhiyun struct work_struct work;
57*4882a593Smuzhiyun struct input_dev *input;
58*4882a593Smuzhiyun struct hid_device *hdev;
59*4882a593Smuzhiyun enum lg_g15_model model;
60*4882a593Smuzhiyun struct lg_g15_led leds[LG_G15_LED_MAX];
61*4882a593Smuzhiyun bool game_mode_enabled;
62*4882a593Smuzhiyun };
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun /******** G15 and G15 v2 LED functions ********/
65*4882a593Smuzhiyun
lg_g15_update_led_brightness(struct lg_g15_data * g15)66*4882a593Smuzhiyun static int lg_g15_update_led_brightness(struct lg_g15_data *g15)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun int ret;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev, LG_G15_FEATURE_REPORT,
71*4882a593Smuzhiyun g15->transfer_buf, 4,
72*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_GET_REPORT);
73*4882a593Smuzhiyun if (ret != 4) {
74*4882a593Smuzhiyun hid_err(g15->hdev, "Error getting LED brightness: %d\n", ret);
75*4882a593Smuzhiyun return (ret < 0) ? ret : -EIO;
76*4882a593Smuzhiyun }
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun g15->leds[LG_G15_KBD_BRIGHTNESS].brightness = g15->transfer_buf[1];
79*4882a593Smuzhiyun g15->leds[LG_G15_LCD_BRIGHTNESS].brightness = g15->transfer_buf[2];
80*4882a593Smuzhiyun
81*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET1].brightness =
82*4882a593Smuzhiyun !(g15->transfer_buf[3] & 0x01);
83*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET2].brightness =
84*4882a593Smuzhiyun !(g15->transfer_buf[3] & 0x02);
85*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET3].brightness =
86*4882a593Smuzhiyun !(g15->transfer_buf[3] & 0x04);
87*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_RECORD].brightness =
88*4882a593Smuzhiyun !(g15->transfer_buf[3] & 0x08);
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun return 0;
91*4882a593Smuzhiyun }
92*4882a593Smuzhiyun
lg_g15_led_get(struct led_classdev * led_cdev)93*4882a593Smuzhiyun static enum led_brightness lg_g15_led_get(struct led_classdev *led_cdev)
94*4882a593Smuzhiyun {
95*4882a593Smuzhiyun struct lg_g15_led *g15_led =
96*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
97*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
98*4882a593Smuzhiyun enum led_brightness brightness;
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun mutex_lock(&g15->mutex);
101*4882a593Smuzhiyun lg_g15_update_led_brightness(g15);
102*4882a593Smuzhiyun brightness = g15->leds[g15_led->led].brightness;
103*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun return brightness;
106*4882a593Smuzhiyun }
107*4882a593Smuzhiyun
lg_g15_led_set(struct led_classdev * led_cdev,enum led_brightness brightness)108*4882a593Smuzhiyun static int lg_g15_led_set(struct led_classdev *led_cdev,
109*4882a593Smuzhiyun enum led_brightness brightness)
110*4882a593Smuzhiyun {
111*4882a593Smuzhiyun struct lg_g15_led *g15_led =
112*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
113*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
114*4882a593Smuzhiyun u8 val, mask = 0;
115*4882a593Smuzhiyun int i, ret;
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun /* Ignore LED off on unregister / keyboard unplug */
118*4882a593Smuzhiyun if (led_cdev->flags & LED_UNREGISTERING)
119*4882a593Smuzhiyun return 0;
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun mutex_lock(&g15->mutex);
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun g15->transfer_buf[0] = LG_G15_FEATURE_REPORT;
124*4882a593Smuzhiyun g15->transfer_buf[3] = 0;
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun if (g15_led->led < LG_G15_BRIGHTNESS_MAX) {
127*4882a593Smuzhiyun g15->transfer_buf[1] = g15_led->led + 1;
128*4882a593Smuzhiyun g15->transfer_buf[2] = brightness << (g15_led->led * 4);
129*4882a593Smuzhiyun } else {
130*4882a593Smuzhiyun for (i = LG_G15_MACRO_PRESET1; i < LG_G15_LED_MAX; i++) {
131*4882a593Smuzhiyun if (i == g15_led->led)
132*4882a593Smuzhiyun val = brightness;
133*4882a593Smuzhiyun else
134*4882a593Smuzhiyun val = g15->leds[i].brightness;
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun if (val)
137*4882a593Smuzhiyun mask |= 1 << (i - LG_G15_MACRO_PRESET1);
138*4882a593Smuzhiyun }
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun g15->transfer_buf[1] = 0x04;
141*4882a593Smuzhiyun g15->transfer_buf[2] = ~mask;
142*4882a593Smuzhiyun }
143*4882a593Smuzhiyun
144*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev, LG_G15_FEATURE_REPORT,
145*4882a593Smuzhiyun g15->transfer_buf, 4,
146*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_SET_REPORT);
147*4882a593Smuzhiyun if (ret == 4) {
148*4882a593Smuzhiyun /* Success */
149*4882a593Smuzhiyun g15_led->brightness = brightness;
150*4882a593Smuzhiyun ret = 0;
151*4882a593Smuzhiyun } else {
152*4882a593Smuzhiyun hid_err(g15->hdev, "Error setting LED brightness: %d\n", ret);
153*4882a593Smuzhiyun ret = (ret < 0) ? ret : -EIO;
154*4882a593Smuzhiyun }
155*4882a593Smuzhiyun
156*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun return ret;
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun
lg_g15_leds_changed_work(struct work_struct * work)161*4882a593Smuzhiyun static void lg_g15_leds_changed_work(struct work_struct *work)
162*4882a593Smuzhiyun {
163*4882a593Smuzhiyun struct lg_g15_data *g15 = container_of(work, struct lg_g15_data, work);
164*4882a593Smuzhiyun enum led_brightness old_brightness[LG_G15_BRIGHTNESS_MAX];
165*4882a593Smuzhiyun enum led_brightness brightness[LG_G15_BRIGHTNESS_MAX];
166*4882a593Smuzhiyun int i, ret;
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun mutex_lock(&g15->mutex);
169*4882a593Smuzhiyun for (i = 0; i < LG_G15_BRIGHTNESS_MAX; i++)
170*4882a593Smuzhiyun old_brightness[i] = g15->leds[i].brightness;
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun ret = lg_g15_update_led_brightness(g15);
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun for (i = 0; i < LG_G15_BRIGHTNESS_MAX; i++)
175*4882a593Smuzhiyun brightness[i] = g15->leds[i].brightness;
176*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
177*4882a593Smuzhiyun
178*4882a593Smuzhiyun if (ret)
179*4882a593Smuzhiyun return;
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun for (i = 0; i < LG_G15_BRIGHTNESS_MAX; i++) {
182*4882a593Smuzhiyun if (brightness[i] == old_brightness[i])
183*4882a593Smuzhiyun continue;
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun led_classdev_notify_brightness_hw_changed(&g15->leds[i].cdev,
186*4882a593Smuzhiyun brightness[i]);
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun }
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun /******** G510 LED functions ********/
191*4882a593Smuzhiyun
lg_g510_get_initial_led_brightness(struct lg_g15_data * g15,int i)192*4882a593Smuzhiyun static int lg_g510_get_initial_led_brightness(struct lg_g15_data *g15, int i)
193*4882a593Smuzhiyun {
194*4882a593Smuzhiyun int ret, high;
195*4882a593Smuzhiyun
196*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev, LG_G510_FEATURE_BACKLIGHT_RGB + i,
197*4882a593Smuzhiyun g15->transfer_buf, 4,
198*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_GET_REPORT);
199*4882a593Smuzhiyun if (ret != 4) {
200*4882a593Smuzhiyun hid_err(g15->hdev, "Error getting LED brightness: %d\n", ret);
201*4882a593Smuzhiyun return (ret < 0) ? ret : -EIO;
202*4882a593Smuzhiyun }
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun high = max3(g15->transfer_buf[1], g15->transfer_buf[2],
205*4882a593Smuzhiyun g15->transfer_buf[3]);
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun if (high) {
208*4882a593Smuzhiyun g15->leds[i].red =
209*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15->transfer_buf[1] * 255, high);
210*4882a593Smuzhiyun g15->leds[i].green =
211*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15->transfer_buf[2] * 255, high);
212*4882a593Smuzhiyun g15->leds[i].blue =
213*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15->transfer_buf[3] * 255, high);
214*4882a593Smuzhiyun g15->leds[i].brightness = high;
215*4882a593Smuzhiyun } else {
216*4882a593Smuzhiyun g15->leds[i].red = 255;
217*4882a593Smuzhiyun g15->leds[i].green = 255;
218*4882a593Smuzhiyun g15->leds[i].blue = 255;
219*4882a593Smuzhiyun g15->leds[i].brightness = 0;
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun return 0;
223*4882a593Smuzhiyun }
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun /* Must be called with g15->mutex locked */
lg_g510_kbd_led_write(struct lg_g15_data * g15,struct lg_g15_led * g15_led,enum led_brightness brightness)226*4882a593Smuzhiyun static int lg_g510_kbd_led_write(struct lg_g15_data *g15,
227*4882a593Smuzhiyun struct lg_g15_led *g15_led,
228*4882a593Smuzhiyun enum led_brightness brightness)
229*4882a593Smuzhiyun {
230*4882a593Smuzhiyun int ret;
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun g15->transfer_buf[0] = 5 + g15_led->led;
233*4882a593Smuzhiyun g15->transfer_buf[1] =
234*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15_led->red * brightness, 255);
235*4882a593Smuzhiyun g15->transfer_buf[2] =
236*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15_led->green * brightness, 255);
237*4882a593Smuzhiyun g15->transfer_buf[3] =
238*4882a593Smuzhiyun DIV_ROUND_CLOSEST(g15_led->blue * brightness, 255);
239*4882a593Smuzhiyun
240*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev,
241*4882a593Smuzhiyun LG_G510_FEATURE_BACKLIGHT_RGB + g15_led->led,
242*4882a593Smuzhiyun g15->transfer_buf, 4,
243*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_SET_REPORT);
244*4882a593Smuzhiyun if (ret == 4) {
245*4882a593Smuzhiyun /* Success */
246*4882a593Smuzhiyun g15_led->brightness = brightness;
247*4882a593Smuzhiyun ret = 0;
248*4882a593Smuzhiyun } else {
249*4882a593Smuzhiyun hid_err(g15->hdev, "Error setting LED brightness: %d\n", ret);
250*4882a593Smuzhiyun ret = (ret < 0) ? ret : -EIO;
251*4882a593Smuzhiyun }
252*4882a593Smuzhiyun
253*4882a593Smuzhiyun return ret;
254*4882a593Smuzhiyun }
255*4882a593Smuzhiyun
lg_g510_kbd_led_set(struct led_classdev * led_cdev,enum led_brightness brightness)256*4882a593Smuzhiyun static int lg_g510_kbd_led_set(struct led_classdev *led_cdev,
257*4882a593Smuzhiyun enum led_brightness brightness)
258*4882a593Smuzhiyun {
259*4882a593Smuzhiyun struct lg_g15_led *g15_led =
260*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
261*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
262*4882a593Smuzhiyun int ret;
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun /* Ignore LED off on unregister / keyboard unplug */
265*4882a593Smuzhiyun if (led_cdev->flags & LED_UNREGISTERING)
266*4882a593Smuzhiyun return 0;
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun mutex_lock(&g15->mutex);
269*4882a593Smuzhiyun ret = lg_g510_kbd_led_write(g15, g15_led, brightness);
270*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun return ret;
273*4882a593Smuzhiyun }
274*4882a593Smuzhiyun
lg_g510_kbd_led_get(struct led_classdev * led_cdev)275*4882a593Smuzhiyun static enum led_brightness lg_g510_kbd_led_get(struct led_classdev *led_cdev)
276*4882a593Smuzhiyun {
277*4882a593Smuzhiyun struct lg_g15_led *g15_led =
278*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun return g15_led->brightness;
281*4882a593Smuzhiyun }
282*4882a593Smuzhiyun
color_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)283*4882a593Smuzhiyun static ssize_t color_store(struct device *dev, struct device_attribute *attr,
284*4882a593Smuzhiyun const char *buf, size_t count)
285*4882a593Smuzhiyun {
286*4882a593Smuzhiyun struct led_classdev *led_cdev = dev_get_drvdata(dev);
287*4882a593Smuzhiyun struct lg_g15_led *g15_led =
288*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
289*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
290*4882a593Smuzhiyun unsigned long value;
291*4882a593Smuzhiyun int ret;
292*4882a593Smuzhiyun
293*4882a593Smuzhiyun if (count < 7 || (count == 8 && buf[7] != '\n') || count > 8)
294*4882a593Smuzhiyun return -EINVAL;
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun if (buf[0] != '#')
297*4882a593Smuzhiyun return -EINVAL;
298*4882a593Smuzhiyun
299*4882a593Smuzhiyun ret = kstrtoul(buf + 1, 16, &value);
300*4882a593Smuzhiyun if (ret)
301*4882a593Smuzhiyun return ret;
302*4882a593Smuzhiyun
303*4882a593Smuzhiyun mutex_lock(&g15->mutex);
304*4882a593Smuzhiyun g15_led->red = (value & 0xff0000) >> 16;
305*4882a593Smuzhiyun g15_led->green = (value & 0x00ff00) >> 8;
306*4882a593Smuzhiyun g15_led->blue = (value & 0x0000ff);
307*4882a593Smuzhiyun ret = lg_g510_kbd_led_write(g15, g15_led, g15_led->brightness);
308*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun return (ret < 0) ? ret : count;
311*4882a593Smuzhiyun }
312*4882a593Smuzhiyun
color_show(struct device * dev,struct device_attribute * attr,char * buf)313*4882a593Smuzhiyun static ssize_t color_show(struct device *dev, struct device_attribute *attr,
314*4882a593Smuzhiyun char *buf)
315*4882a593Smuzhiyun {
316*4882a593Smuzhiyun struct led_classdev *led_cdev = dev_get_drvdata(dev);
317*4882a593Smuzhiyun struct lg_g15_led *g15_led =
318*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
319*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
320*4882a593Smuzhiyun ssize_t ret;
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun mutex_lock(&g15->mutex);
323*4882a593Smuzhiyun ret = sprintf(buf, "#%02x%02x%02x\n",
324*4882a593Smuzhiyun g15_led->red, g15_led->green, g15_led->blue);
325*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
326*4882a593Smuzhiyun
327*4882a593Smuzhiyun return ret;
328*4882a593Smuzhiyun }
329*4882a593Smuzhiyun
330*4882a593Smuzhiyun static DEVICE_ATTR_RW(color);
331*4882a593Smuzhiyun
332*4882a593Smuzhiyun static struct attribute *lg_g510_kbd_led_attrs[] = {
333*4882a593Smuzhiyun &dev_attr_color.attr,
334*4882a593Smuzhiyun NULL,
335*4882a593Smuzhiyun };
336*4882a593Smuzhiyun
337*4882a593Smuzhiyun static const struct attribute_group lg_g510_kbd_led_group = {
338*4882a593Smuzhiyun .attrs = lg_g510_kbd_led_attrs,
339*4882a593Smuzhiyun };
340*4882a593Smuzhiyun
341*4882a593Smuzhiyun static const struct attribute_group *lg_g510_kbd_led_groups[] = {
342*4882a593Smuzhiyun &lg_g510_kbd_led_group,
343*4882a593Smuzhiyun NULL,
344*4882a593Smuzhiyun };
345*4882a593Smuzhiyun
lg_g510_leds_sync_work(struct work_struct * work)346*4882a593Smuzhiyun static void lg_g510_leds_sync_work(struct work_struct *work)
347*4882a593Smuzhiyun {
348*4882a593Smuzhiyun struct lg_g15_data *g15 = container_of(work, struct lg_g15_data, work);
349*4882a593Smuzhiyun
350*4882a593Smuzhiyun mutex_lock(&g15->mutex);
351*4882a593Smuzhiyun lg_g510_kbd_led_write(g15, &g15->leds[LG_G15_KBD_BRIGHTNESS],
352*4882a593Smuzhiyun g15->leds[LG_G15_KBD_BRIGHTNESS].brightness);
353*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
354*4882a593Smuzhiyun }
355*4882a593Smuzhiyun
lg_g510_update_mkey_led_brightness(struct lg_g15_data * g15)356*4882a593Smuzhiyun static int lg_g510_update_mkey_led_brightness(struct lg_g15_data *g15)
357*4882a593Smuzhiyun {
358*4882a593Smuzhiyun int ret;
359*4882a593Smuzhiyun
360*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev, LG_G510_FEATURE_M_KEYS_LEDS,
361*4882a593Smuzhiyun g15->transfer_buf, 2,
362*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_GET_REPORT);
363*4882a593Smuzhiyun if (ret != 2) {
364*4882a593Smuzhiyun hid_err(g15->hdev, "Error getting LED brightness: %d\n", ret);
365*4882a593Smuzhiyun ret = (ret < 0) ? ret : -EIO;
366*4882a593Smuzhiyun }
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET1].brightness =
369*4882a593Smuzhiyun !!(g15->transfer_buf[1] & 0x80);
370*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET2].brightness =
371*4882a593Smuzhiyun !!(g15->transfer_buf[1] & 0x40);
372*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_PRESET3].brightness =
373*4882a593Smuzhiyun !!(g15->transfer_buf[1] & 0x20);
374*4882a593Smuzhiyun g15->leds[LG_G15_MACRO_RECORD].brightness =
375*4882a593Smuzhiyun !!(g15->transfer_buf[1] & 0x10);
376*4882a593Smuzhiyun
377*4882a593Smuzhiyun return 0;
378*4882a593Smuzhiyun }
379*4882a593Smuzhiyun
lg_g510_mkey_led_get(struct led_classdev * led_cdev)380*4882a593Smuzhiyun static enum led_brightness lg_g510_mkey_led_get(struct led_classdev *led_cdev)
381*4882a593Smuzhiyun {
382*4882a593Smuzhiyun struct lg_g15_led *g15_led =
383*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
384*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
385*4882a593Smuzhiyun enum led_brightness brightness;
386*4882a593Smuzhiyun
387*4882a593Smuzhiyun mutex_lock(&g15->mutex);
388*4882a593Smuzhiyun lg_g510_update_mkey_led_brightness(g15);
389*4882a593Smuzhiyun brightness = g15->leds[g15_led->led].brightness;
390*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
391*4882a593Smuzhiyun
392*4882a593Smuzhiyun return brightness;
393*4882a593Smuzhiyun }
394*4882a593Smuzhiyun
lg_g510_mkey_led_set(struct led_classdev * led_cdev,enum led_brightness brightness)395*4882a593Smuzhiyun static int lg_g510_mkey_led_set(struct led_classdev *led_cdev,
396*4882a593Smuzhiyun enum led_brightness brightness)
397*4882a593Smuzhiyun {
398*4882a593Smuzhiyun struct lg_g15_led *g15_led =
399*4882a593Smuzhiyun container_of(led_cdev, struct lg_g15_led, cdev);
400*4882a593Smuzhiyun struct lg_g15_data *g15 = dev_get_drvdata(led_cdev->dev->parent);
401*4882a593Smuzhiyun u8 val, mask = 0;
402*4882a593Smuzhiyun int i, ret;
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun /* Ignore LED off on unregister / keyboard unplug */
405*4882a593Smuzhiyun if (led_cdev->flags & LED_UNREGISTERING)
406*4882a593Smuzhiyun return 0;
407*4882a593Smuzhiyun
408*4882a593Smuzhiyun mutex_lock(&g15->mutex);
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun for (i = LG_G15_MACRO_PRESET1; i < LG_G15_LED_MAX; i++) {
411*4882a593Smuzhiyun if (i == g15_led->led)
412*4882a593Smuzhiyun val = brightness;
413*4882a593Smuzhiyun else
414*4882a593Smuzhiyun val = g15->leds[i].brightness;
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun if (val)
417*4882a593Smuzhiyun mask |= 0x80 >> (i - LG_G15_MACRO_PRESET1);
418*4882a593Smuzhiyun }
419*4882a593Smuzhiyun
420*4882a593Smuzhiyun g15->transfer_buf[0] = LG_G510_FEATURE_M_KEYS_LEDS;
421*4882a593Smuzhiyun g15->transfer_buf[1] = mask;
422*4882a593Smuzhiyun
423*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev, LG_G510_FEATURE_M_KEYS_LEDS,
424*4882a593Smuzhiyun g15->transfer_buf, 2,
425*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_SET_REPORT);
426*4882a593Smuzhiyun if (ret == 2) {
427*4882a593Smuzhiyun /* Success */
428*4882a593Smuzhiyun g15_led->brightness = brightness;
429*4882a593Smuzhiyun ret = 0;
430*4882a593Smuzhiyun } else {
431*4882a593Smuzhiyun hid_err(g15->hdev, "Error setting LED brightness: %d\n", ret);
432*4882a593Smuzhiyun ret = (ret < 0) ? ret : -EIO;
433*4882a593Smuzhiyun }
434*4882a593Smuzhiyun
435*4882a593Smuzhiyun mutex_unlock(&g15->mutex);
436*4882a593Smuzhiyun
437*4882a593Smuzhiyun return ret;
438*4882a593Smuzhiyun }
439*4882a593Smuzhiyun
440*4882a593Smuzhiyun /******** Generic LED functions ********/
lg_g15_get_initial_led_brightness(struct lg_g15_data * g15)441*4882a593Smuzhiyun static int lg_g15_get_initial_led_brightness(struct lg_g15_data *g15)
442*4882a593Smuzhiyun {
443*4882a593Smuzhiyun int ret;
444*4882a593Smuzhiyun
445*4882a593Smuzhiyun switch (g15->model) {
446*4882a593Smuzhiyun case LG_G15:
447*4882a593Smuzhiyun case LG_G15_V2:
448*4882a593Smuzhiyun return lg_g15_update_led_brightness(g15);
449*4882a593Smuzhiyun case LG_G510:
450*4882a593Smuzhiyun case LG_G510_USB_AUDIO:
451*4882a593Smuzhiyun ret = lg_g510_get_initial_led_brightness(g15, 0);
452*4882a593Smuzhiyun if (ret)
453*4882a593Smuzhiyun return ret;
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun ret = lg_g510_get_initial_led_brightness(g15, 1);
456*4882a593Smuzhiyun if (ret)
457*4882a593Smuzhiyun return ret;
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun return lg_g510_update_mkey_led_brightness(g15);
460*4882a593Smuzhiyun }
461*4882a593Smuzhiyun return -EINVAL; /* Never reached */
462*4882a593Smuzhiyun }
463*4882a593Smuzhiyun
464*4882a593Smuzhiyun /******** Input functions ********/
465*4882a593Smuzhiyun
466*4882a593Smuzhiyun /* On the G15 Mark I Logitech has been quite creative with which bit is what */
lg_g15_event(struct lg_g15_data * g15,u8 * data,int size)467*4882a593Smuzhiyun static int lg_g15_event(struct lg_g15_data *g15, u8 *data, int size)
468*4882a593Smuzhiyun {
469*4882a593Smuzhiyun int i, val;
470*4882a593Smuzhiyun
471*4882a593Smuzhiyun /* G1 - G6 */
472*4882a593Smuzhiyun for (i = 0; i < 6; i++) {
473*4882a593Smuzhiyun val = data[i + 1] & (1 << i);
474*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO1 + i, val);
475*4882a593Smuzhiyun }
476*4882a593Smuzhiyun /* G7 - G12 */
477*4882a593Smuzhiyun for (i = 0; i < 6; i++) {
478*4882a593Smuzhiyun val = data[i + 2] & (1 << i);
479*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO7 + i, val);
480*4882a593Smuzhiyun }
481*4882a593Smuzhiyun /* G13 - G17 */
482*4882a593Smuzhiyun for (i = 0; i < 5; i++) {
483*4882a593Smuzhiyun val = data[i + 1] & (4 << i);
484*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO13 + i, val);
485*4882a593Smuzhiyun }
486*4882a593Smuzhiyun /* G18 */
487*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO18, data[8] & 0x40);
488*4882a593Smuzhiyun
489*4882a593Smuzhiyun /* M1 - M3 */
490*4882a593Smuzhiyun for (i = 0; i < 3; i++) {
491*4882a593Smuzhiyun val = data[i + 6] & (1 << i);
492*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_PRESET1 + i, val);
493*4882a593Smuzhiyun }
494*4882a593Smuzhiyun /* MR */
495*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_RECORD_START, data[7] & 0x40);
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun /* Most left (round) button below the LCD */
498*4882a593Smuzhiyun input_report_key(g15->input, KEY_KBD_LCD_MENU1, data[8] & 0x80);
499*4882a593Smuzhiyun /* 4 other buttons below the LCD */
500*4882a593Smuzhiyun for (i = 0; i < 4; i++) {
501*4882a593Smuzhiyun val = data[i + 2] & 0x80;
502*4882a593Smuzhiyun input_report_key(g15->input, KEY_KBD_LCD_MENU2 + i, val);
503*4882a593Smuzhiyun }
504*4882a593Smuzhiyun
505*4882a593Smuzhiyun /* Backlight cycle button pressed? */
506*4882a593Smuzhiyun if (data[1] & 0x80)
507*4882a593Smuzhiyun schedule_work(&g15->work);
508*4882a593Smuzhiyun
509*4882a593Smuzhiyun input_sync(g15->input);
510*4882a593Smuzhiyun return 0;
511*4882a593Smuzhiyun }
512*4882a593Smuzhiyun
lg_g15_v2_event(struct lg_g15_data * g15,u8 * data,int size)513*4882a593Smuzhiyun static int lg_g15_v2_event(struct lg_g15_data *g15, u8 *data, int size)
514*4882a593Smuzhiyun {
515*4882a593Smuzhiyun int i, val;
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun /* G1 - G6 */
518*4882a593Smuzhiyun for (i = 0; i < 6; i++) {
519*4882a593Smuzhiyun val = data[1] & (1 << i);
520*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO1 + i, val);
521*4882a593Smuzhiyun }
522*4882a593Smuzhiyun
523*4882a593Smuzhiyun /* M1 - M3 + MR */
524*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_PRESET1, data[1] & 0x40);
525*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_PRESET2, data[1] & 0x80);
526*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_PRESET3, data[2] & 0x20);
527*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_RECORD_START, data[2] & 0x40);
528*4882a593Smuzhiyun
529*4882a593Smuzhiyun /* Round button to the left of the LCD */
530*4882a593Smuzhiyun input_report_key(g15->input, KEY_KBD_LCD_MENU1, data[2] & 0x80);
531*4882a593Smuzhiyun /* 4 buttons below the LCD */
532*4882a593Smuzhiyun for (i = 0; i < 4; i++) {
533*4882a593Smuzhiyun val = data[2] & (2 << i);
534*4882a593Smuzhiyun input_report_key(g15->input, KEY_KBD_LCD_MENU2 + i, val);
535*4882a593Smuzhiyun }
536*4882a593Smuzhiyun
537*4882a593Smuzhiyun /* Backlight cycle button pressed? */
538*4882a593Smuzhiyun if (data[2] & 0x01)
539*4882a593Smuzhiyun schedule_work(&g15->work);
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun input_sync(g15->input);
542*4882a593Smuzhiyun return 0;
543*4882a593Smuzhiyun }
544*4882a593Smuzhiyun
lg_g510_event(struct lg_g15_data * g15,u8 * data,int size)545*4882a593Smuzhiyun static int lg_g510_event(struct lg_g15_data *g15, u8 *data, int size)
546*4882a593Smuzhiyun {
547*4882a593Smuzhiyun bool game_mode_enabled;
548*4882a593Smuzhiyun int i, val;
549*4882a593Smuzhiyun
550*4882a593Smuzhiyun /* G1 - G18 */
551*4882a593Smuzhiyun for (i = 0; i < 18; i++) {
552*4882a593Smuzhiyun val = data[i / 8 + 1] & (1 << (i % 8));
553*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO1 + i, val);
554*4882a593Smuzhiyun }
555*4882a593Smuzhiyun
556*4882a593Smuzhiyun /* Game mode on/off slider */
557*4882a593Smuzhiyun game_mode_enabled = data[3] & 0x04;
558*4882a593Smuzhiyun if (game_mode_enabled != g15->game_mode_enabled) {
559*4882a593Smuzhiyun if (game_mode_enabled)
560*4882a593Smuzhiyun hid_info(g15->hdev, "Game Mode enabled, Windows (super) key is disabled\n");
561*4882a593Smuzhiyun else
562*4882a593Smuzhiyun hid_info(g15->hdev, "Game Mode disabled\n");
563*4882a593Smuzhiyun g15->game_mode_enabled = game_mode_enabled;
564*4882a593Smuzhiyun }
565*4882a593Smuzhiyun
566*4882a593Smuzhiyun /* M1 - M3 */
567*4882a593Smuzhiyun for (i = 0; i < 3; i++) {
568*4882a593Smuzhiyun val = data[3] & (0x10 << i);
569*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_PRESET1 + i, val);
570*4882a593Smuzhiyun }
571*4882a593Smuzhiyun /* MR */
572*4882a593Smuzhiyun input_report_key(g15->input, KEY_MACRO_RECORD_START, data[3] & 0x80);
573*4882a593Smuzhiyun
574*4882a593Smuzhiyun /* LCD menu keys */
575*4882a593Smuzhiyun for (i = 0; i < 5; i++) {
576*4882a593Smuzhiyun val = data[4] & (1 << i);
577*4882a593Smuzhiyun input_report_key(g15->input, KEY_KBD_LCD_MENU1 + i, val);
578*4882a593Smuzhiyun }
579*4882a593Smuzhiyun
580*4882a593Smuzhiyun /* Headphone Mute */
581*4882a593Smuzhiyun input_report_key(g15->input, KEY_MUTE, data[4] & 0x20);
582*4882a593Smuzhiyun /* Microphone Mute */
583*4882a593Smuzhiyun input_report_key(g15->input, KEY_F20, data[4] & 0x40);
584*4882a593Smuzhiyun
585*4882a593Smuzhiyun input_sync(g15->input);
586*4882a593Smuzhiyun return 0;
587*4882a593Smuzhiyun }
588*4882a593Smuzhiyun
lg_g510_leds_event(struct lg_g15_data * g15,u8 * data,int size)589*4882a593Smuzhiyun static int lg_g510_leds_event(struct lg_g15_data *g15, u8 *data, int size)
590*4882a593Smuzhiyun {
591*4882a593Smuzhiyun bool backlight_disabled;
592*4882a593Smuzhiyun
593*4882a593Smuzhiyun /*
594*4882a593Smuzhiyun * The G510 ignores backlight updates when the backlight is turned off
595*4882a593Smuzhiyun * through the light toggle button on the keyboard, to work around this
596*4882a593Smuzhiyun * we queue a workitem to sync values when the backlight is turned on.
597*4882a593Smuzhiyun */
598*4882a593Smuzhiyun backlight_disabled = data[1] & 0x04;
599*4882a593Smuzhiyun if (!backlight_disabled)
600*4882a593Smuzhiyun schedule_work(&g15->work);
601*4882a593Smuzhiyun
602*4882a593Smuzhiyun return 0;
603*4882a593Smuzhiyun }
604*4882a593Smuzhiyun
lg_g15_raw_event(struct hid_device * hdev,struct hid_report * report,u8 * data,int size)605*4882a593Smuzhiyun static int lg_g15_raw_event(struct hid_device *hdev, struct hid_report *report,
606*4882a593Smuzhiyun u8 *data, int size)
607*4882a593Smuzhiyun {
608*4882a593Smuzhiyun struct lg_g15_data *g15 = hid_get_drvdata(hdev);
609*4882a593Smuzhiyun
610*4882a593Smuzhiyun if (!g15)
611*4882a593Smuzhiyun return 0;
612*4882a593Smuzhiyun
613*4882a593Smuzhiyun switch (g15->model) {
614*4882a593Smuzhiyun case LG_G15:
615*4882a593Smuzhiyun if (data[0] == 0x02 && size == 9)
616*4882a593Smuzhiyun return lg_g15_event(g15, data, size);
617*4882a593Smuzhiyun break;
618*4882a593Smuzhiyun case LG_G15_V2:
619*4882a593Smuzhiyun if (data[0] == 0x02 && size == 5)
620*4882a593Smuzhiyun return lg_g15_v2_event(g15, data, size);
621*4882a593Smuzhiyun break;
622*4882a593Smuzhiyun case LG_G510:
623*4882a593Smuzhiyun case LG_G510_USB_AUDIO:
624*4882a593Smuzhiyun if (data[0] == 0x03 && size == 5)
625*4882a593Smuzhiyun return lg_g510_event(g15, data, size);
626*4882a593Smuzhiyun if (data[0] == 0x04 && size == 2)
627*4882a593Smuzhiyun return lg_g510_leds_event(g15, data, size);
628*4882a593Smuzhiyun break;
629*4882a593Smuzhiyun }
630*4882a593Smuzhiyun
631*4882a593Smuzhiyun return 0;
632*4882a593Smuzhiyun }
633*4882a593Smuzhiyun
lg_g15_input_open(struct input_dev * dev)634*4882a593Smuzhiyun static int lg_g15_input_open(struct input_dev *dev)
635*4882a593Smuzhiyun {
636*4882a593Smuzhiyun struct hid_device *hdev = input_get_drvdata(dev);
637*4882a593Smuzhiyun
638*4882a593Smuzhiyun return hid_hw_open(hdev);
639*4882a593Smuzhiyun }
640*4882a593Smuzhiyun
lg_g15_input_close(struct input_dev * dev)641*4882a593Smuzhiyun static void lg_g15_input_close(struct input_dev *dev)
642*4882a593Smuzhiyun {
643*4882a593Smuzhiyun struct hid_device *hdev = input_get_drvdata(dev);
644*4882a593Smuzhiyun
645*4882a593Smuzhiyun hid_hw_close(hdev);
646*4882a593Smuzhiyun }
647*4882a593Smuzhiyun
lg_g15_register_led(struct lg_g15_data * g15,int i)648*4882a593Smuzhiyun static int lg_g15_register_led(struct lg_g15_data *g15, int i)
649*4882a593Smuzhiyun {
650*4882a593Smuzhiyun const char * const led_names[] = {
651*4882a593Smuzhiyun "g15::kbd_backlight",
652*4882a593Smuzhiyun "g15::lcd_backlight",
653*4882a593Smuzhiyun "g15::macro_preset1",
654*4882a593Smuzhiyun "g15::macro_preset2",
655*4882a593Smuzhiyun "g15::macro_preset3",
656*4882a593Smuzhiyun "g15::macro_record",
657*4882a593Smuzhiyun };
658*4882a593Smuzhiyun
659*4882a593Smuzhiyun g15->leds[i].led = i;
660*4882a593Smuzhiyun g15->leds[i].cdev.name = led_names[i];
661*4882a593Smuzhiyun
662*4882a593Smuzhiyun switch (g15->model) {
663*4882a593Smuzhiyun case LG_G15:
664*4882a593Smuzhiyun case LG_G15_V2:
665*4882a593Smuzhiyun g15->leds[i].cdev.brightness_set_blocking = lg_g15_led_set;
666*4882a593Smuzhiyun g15->leds[i].cdev.brightness_get = lg_g15_led_get;
667*4882a593Smuzhiyun if (i < LG_G15_BRIGHTNESS_MAX) {
668*4882a593Smuzhiyun g15->leds[i].cdev.flags = LED_BRIGHT_HW_CHANGED;
669*4882a593Smuzhiyun g15->leds[i].cdev.max_brightness = 2;
670*4882a593Smuzhiyun } else {
671*4882a593Smuzhiyun g15->leds[i].cdev.max_brightness = 1;
672*4882a593Smuzhiyun }
673*4882a593Smuzhiyun break;
674*4882a593Smuzhiyun case LG_G510:
675*4882a593Smuzhiyun case LG_G510_USB_AUDIO:
676*4882a593Smuzhiyun switch (i) {
677*4882a593Smuzhiyun case LG_G15_LCD_BRIGHTNESS:
678*4882a593Smuzhiyun /*
679*4882a593Smuzhiyun * The G510 does not have a separate LCD brightness,
680*4882a593Smuzhiyun * but it does have a separate power-on (reset) value.
681*4882a593Smuzhiyun */
682*4882a593Smuzhiyun g15->leds[i].cdev.name = "g15::power_on_backlight_val";
683*4882a593Smuzhiyun fallthrough;
684*4882a593Smuzhiyun case LG_G15_KBD_BRIGHTNESS:
685*4882a593Smuzhiyun g15->leds[i].cdev.brightness_set_blocking =
686*4882a593Smuzhiyun lg_g510_kbd_led_set;
687*4882a593Smuzhiyun g15->leds[i].cdev.brightness_get =
688*4882a593Smuzhiyun lg_g510_kbd_led_get;
689*4882a593Smuzhiyun g15->leds[i].cdev.max_brightness = 255;
690*4882a593Smuzhiyun g15->leds[i].cdev.groups = lg_g510_kbd_led_groups;
691*4882a593Smuzhiyun break;
692*4882a593Smuzhiyun default:
693*4882a593Smuzhiyun g15->leds[i].cdev.brightness_set_blocking =
694*4882a593Smuzhiyun lg_g510_mkey_led_set;
695*4882a593Smuzhiyun g15->leds[i].cdev.brightness_get =
696*4882a593Smuzhiyun lg_g510_mkey_led_get;
697*4882a593Smuzhiyun g15->leds[i].cdev.max_brightness = 1;
698*4882a593Smuzhiyun }
699*4882a593Smuzhiyun break;
700*4882a593Smuzhiyun }
701*4882a593Smuzhiyun
702*4882a593Smuzhiyun return devm_led_classdev_register(&g15->hdev->dev, &g15->leds[i].cdev);
703*4882a593Smuzhiyun }
704*4882a593Smuzhiyun
lg_g15_probe(struct hid_device * hdev,const struct hid_device_id * id)705*4882a593Smuzhiyun static int lg_g15_probe(struct hid_device *hdev, const struct hid_device_id *id)
706*4882a593Smuzhiyun {
707*4882a593Smuzhiyun u8 gkeys_settings_output_report = 0;
708*4882a593Smuzhiyun u8 gkeys_settings_feature_report = 0;
709*4882a593Smuzhiyun struct hid_report_enum *rep_enum;
710*4882a593Smuzhiyun unsigned int connect_mask = 0;
711*4882a593Smuzhiyun bool has_ff000000 = false;
712*4882a593Smuzhiyun struct lg_g15_data *g15;
713*4882a593Smuzhiyun struct input_dev *input;
714*4882a593Smuzhiyun struct hid_report *rep;
715*4882a593Smuzhiyun int ret, i, gkeys = 0;
716*4882a593Smuzhiyun
717*4882a593Smuzhiyun hdev->quirks |= HID_QUIRK_INPUT_PER_APP;
718*4882a593Smuzhiyun
719*4882a593Smuzhiyun ret = hid_parse(hdev);
720*4882a593Smuzhiyun if (ret)
721*4882a593Smuzhiyun return ret;
722*4882a593Smuzhiyun
723*4882a593Smuzhiyun /*
724*4882a593Smuzhiyun * Some models have multiple interfaces, we want the interface with
725*4882a593Smuzhiyun * with the f000.0000 application input report.
726*4882a593Smuzhiyun */
727*4882a593Smuzhiyun rep_enum = &hdev->report_enum[HID_INPUT_REPORT];
728*4882a593Smuzhiyun list_for_each_entry(rep, &rep_enum->report_list, list) {
729*4882a593Smuzhiyun if (rep->application == 0xff000000)
730*4882a593Smuzhiyun has_ff000000 = true;
731*4882a593Smuzhiyun }
732*4882a593Smuzhiyun if (!has_ff000000)
733*4882a593Smuzhiyun return hid_hw_start(hdev, HID_CONNECT_DEFAULT);
734*4882a593Smuzhiyun
735*4882a593Smuzhiyun g15 = devm_kzalloc(&hdev->dev, sizeof(*g15), GFP_KERNEL);
736*4882a593Smuzhiyun if (!g15)
737*4882a593Smuzhiyun return -ENOMEM;
738*4882a593Smuzhiyun
739*4882a593Smuzhiyun mutex_init(&g15->mutex);
740*4882a593Smuzhiyun
741*4882a593Smuzhiyun input = devm_input_allocate_device(&hdev->dev);
742*4882a593Smuzhiyun if (!input)
743*4882a593Smuzhiyun return -ENOMEM;
744*4882a593Smuzhiyun
745*4882a593Smuzhiyun g15->hdev = hdev;
746*4882a593Smuzhiyun g15->model = id->driver_data;
747*4882a593Smuzhiyun hid_set_drvdata(hdev, (void *)g15);
748*4882a593Smuzhiyun
749*4882a593Smuzhiyun switch (g15->model) {
750*4882a593Smuzhiyun case LG_G15:
751*4882a593Smuzhiyun INIT_WORK(&g15->work, lg_g15_leds_changed_work);
752*4882a593Smuzhiyun /*
753*4882a593Smuzhiyun * The G15 and G15 v2 use a separate usb-device (on a builtin
754*4882a593Smuzhiyun * hub) which emulates a keyboard for the F1 - F12 emulation
755*4882a593Smuzhiyun * on the G-keys, which we disable, rendering the emulated kbd
756*4882a593Smuzhiyun * non-functional, so we do not let hid-input connect.
757*4882a593Smuzhiyun */
758*4882a593Smuzhiyun connect_mask = HID_CONNECT_HIDRAW;
759*4882a593Smuzhiyun gkeys_settings_output_report = 0x02;
760*4882a593Smuzhiyun gkeys = 18;
761*4882a593Smuzhiyun break;
762*4882a593Smuzhiyun case LG_G15_V2:
763*4882a593Smuzhiyun INIT_WORK(&g15->work, lg_g15_leds_changed_work);
764*4882a593Smuzhiyun connect_mask = HID_CONNECT_HIDRAW;
765*4882a593Smuzhiyun gkeys_settings_output_report = 0x02;
766*4882a593Smuzhiyun gkeys = 6;
767*4882a593Smuzhiyun break;
768*4882a593Smuzhiyun case LG_G510:
769*4882a593Smuzhiyun case LG_G510_USB_AUDIO:
770*4882a593Smuzhiyun INIT_WORK(&g15->work, lg_g510_leds_sync_work);
771*4882a593Smuzhiyun connect_mask = HID_CONNECT_HIDINPUT | HID_CONNECT_HIDRAW;
772*4882a593Smuzhiyun gkeys_settings_feature_report = 0x01;
773*4882a593Smuzhiyun gkeys = 18;
774*4882a593Smuzhiyun break;
775*4882a593Smuzhiyun }
776*4882a593Smuzhiyun
777*4882a593Smuzhiyun ret = hid_hw_start(hdev, connect_mask);
778*4882a593Smuzhiyun if (ret)
779*4882a593Smuzhiyun return ret;
780*4882a593Smuzhiyun
781*4882a593Smuzhiyun /* Tell the keyboard to stop sending F1-F12 + 1-6 for G1 - G18 */
782*4882a593Smuzhiyun if (gkeys_settings_output_report) {
783*4882a593Smuzhiyun g15->transfer_buf[0] = gkeys_settings_output_report;
784*4882a593Smuzhiyun memset(g15->transfer_buf + 1, 0, gkeys);
785*4882a593Smuzhiyun /*
786*4882a593Smuzhiyun * The kbd ignores our output report if we do not queue
787*4882a593Smuzhiyun * an URB on the USB input endpoint first...
788*4882a593Smuzhiyun */
789*4882a593Smuzhiyun ret = hid_hw_open(hdev);
790*4882a593Smuzhiyun if (ret)
791*4882a593Smuzhiyun goto error_hw_stop;
792*4882a593Smuzhiyun ret = hid_hw_output_report(hdev, g15->transfer_buf, gkeys + 1);
793*4882a593Smuzhiyun hid_hw_close(hdev);
794*4882a593Smuzhiyun }
795*4882a593Smuzhiyun
796*4882a593Smuzhiyun if (gkeys_settings_feature_report) {
797*4882a593Smuzhiyun g15->transfer_buf[0] = gkeys_settings_feature_report;
798*4882a593Smuzhiyun memset(g15->transfer_buf + 1, 0, gkeys);
799*4882a593Smuzhiyun ret = hid_hw_raw_request(g15->hdev,
800*4882a593Smuzhiyun gkeys_settings_feature_report,
801*4882a593Smuzhiyun g15->transfer_buf, gkeys + 1,
802*4882a593Smuzhiyun HID_FEATURE_REPORT, HID_REQ_SET_REPORT);
803*4882a593Smuzhiyun }
804*4882a593Smuzhiyun
805*4882a593Smuzhiyun if (ret < 0) {
806*4882a593Smuzhiyun hid_err(hdev, "Error %d disabling keyboard emulation for the G-keys, falling back to generic hid-input driver\n",
807*4882a593Smuzhiyun ret);
808*4882a593Smuzhiyun hid_set_drvdata(hdev, NULL);
809*4882a593Smuzhiyun return 0;
810*4882a593Smuzhiyun }
811*4882a593Smuzhiyun
812*4882a593Smuzhiyun /* Get initial brightness levels */
813*4882a593Smuzhiyun ret = lg_g15_get_initial_led_brightness(g15);
814*4882a593Smuzhiyun if (ret)
815*4882a593Smuzhiyun goto error_hw_stop;
816*4882a593Smuzhiyun
817*4882a593Smuzhiyun /* Setup and register input device */
818*4882a593Smuzhiyun input->name = "Logitech Gaming Keyboard Gaming Keys";
819*4882a593Smuzhiyun input->phys = hdev->phys;
820*4882a593Smuzhiyun input->uniq = hdev->uniq;
821*4882a593Smuzhiyun input->id.bustype = hdev->bus;
822*4882a593Smuzhiyun input->id.vendor = hdev->vendor;
823*4882a593Smuzhiyun input->id.product = hdev->product;
824*4882a593Smuzhiyun input->id.version = hdev->version;
825*4882a593Smuzhiyun input->dev.parent = &hdev->dev;
826*4882a593Smuzhiyun input->open = lg_g15_input_open;
827*4882a593Smuzhiyun input->close = lg_g15_input_close;
828*4882a593Smuzhiyun
829*4882a593Smuzhiyun /* G-keys */
830*4882a593Smuzhiyun for (i = 0; i < gkeys; i++)
831*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_MACRO1 + i);
832*4882a593Smuzhiyun
833*4882a593Smuzhiyun /* M1 - M3 and MR keys */
834*4882a593Smuzhiyun for (i = 0; i < 3; i++)
835*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_MACRO_PRESET1 + i);
836*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_MACRO_RECORD_START);
837*4882a593Smuzhiyun
838*4882a593Smuzhiyun /* Keys below the LCD, intended for controlling a menu on the LCD */
839*4882a593Smuzhiyun for (i = 0; i < 5; i++)
840*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_KBD_LCD_MENU1 + i);
841*4882a593Smuzhiyun
842*4882a593Smuzhiyun /*
843*4882a593Smuzhiyun * On the G510 only report headphone and mic mute keys when *not* using
844*4882a593Smuzhiyun * the builtin USB audio device. When the builtin audio is used these
845*4882a593Smuzhiyun * keys directly toggle mute (and the LEDs) on/off.
846*4882a593Smuzhiyun */
847*4882a593Smuzhiyun if (g15->model == LG_G510) {
848*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_MUTE);
849*4882a593Smuzhiyun /* Userspace expects F20 for micmute */
850*4882a593Smuzhiyun input_set_capability(input, EV_KEY, KEY_F20);
851*4882a593Smuzhiyun }
852*4882a593Smuzhiyun
853*4882a593Smuzhiyun g15->input = input;
854*4882a593Smuzhiyun input_set_drvdata(input, hdev);
855*4882a593Smuzhiyun
856*4882a593Smuzhiyun ret = input_register_device(input);
857*4882a593Smuzhiyun if (ret)
858*4882a593Smuzhiyun goto error_hw_stop;
859*4882a593Smuzhiyun
860*4882a593Smuzhiyun /* Register LED devices */
861*4882a593Smuzhiyun for (i = 0; i < LG_G15_LED_MAX; i++) {
862*4882a593Smuzhiyun ret = lg_g15_register_led(g15, i);
863*4882a593Smuzhiyun if (ret)
864*4882a593Smuzhiyun goto error_hw_stop;
865*4882a593Smuzhiyun }
866*4882a593Smuzhiyun
867*4882a593Smuzhiyun return 0;
868*4882a593Smuzhiyun
869*4882a593Smuzhiyun error_hw_stop:
870*4882a593Smuzhiyun hid_hw_stop(hdev);
871*4882a593Smuzhiyun return ret;
872*4882a593Smuzhiyun }
873*4882a593Smuzhiyun
874*4882a593Smuzhiyun static const struct hid_device_id lg_g15_devices[] = {
875*4882a593Smuzhiyun /* The G11 is a G15 without the LCD, treat it as a G15 */
876*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH,
877*4882a593Smuzhiyun USB_DEVICE_ID_LOGITECH_G11),
878*4882a593Smuzhiyun .driver_data = LG_G15 },
879*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH,
880*4882a593Smuzhiyun USB_DEVICE_ID_LOGITECH_G15_LCD),
881*4882a593Smuzhiyun .driver_data = LG_G15 },
882*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH,
883*4882a593Smuzhiyun USB_DEVICE_ID_LOGITECH_G15_V2_LCD),
884*4882a593Smuzhiyun .driver_data = LG_G15_V2 },
885*4882a593Smuzhiyun /* G510 without a headset plugged in */
886*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH,
887*4882a593Smuzhiyun USB_DEVICE_ID_LOGITECH_G510),
888*4882a593Smuzhiyun .driver_data = LG_G510 },
889*4882a593Smuzhiyun /* G510 with headset plugged in / with extra USB audio interface */
890*4882a593Smuzhiyun { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH,
891*4882a593Smuzhiyun USB_DEVICE_ID_LOGITECH_G510_USB_AUDIO),
892*4882a593Smuzhiyun .driver_data = LG_G510_USB_AUDIO },
893*4882a593Smuzhiyun { }
894*4882a593Smuzhiyun };
895*4882a593Smuzhiyun MODULE_DEVICE_TABLE(hid, lg_g15_devices);
896*4882a593Smuzhiyun
897*4882a593Smuzhiyun static struct hid_driver lg_g15_driver = {
898*4882a593Smuzhiyun .name = "lg-g15",
899*4882a593Smuzhiyun .id_table = lg_g15_devices,
900*4882a593Smuzhiyun .raw_event = lg_g15_raw_event,
901*4882a593Smuzhiyun .probe = lg_g15_probe,
902*4882a593Smuzhiyun };
903*4882a593Smuzhiyun module_hid_driver(lg_g15_driver);
904*4882a593Smuzhiyun
905*4882a593Smuzhiyun MODULE_LICENSE("GPL");
906