1 // SPDX-License-Identifier: GPL-2.0+
3 * HID driver for gaming keys on Logitech gaming keyboards (such as the G15)
5 * Copyright (c) 2019 Hans de Goede <hdegoede@redhat.com>
8 #include <linux/device.h>
10 #include <linux/module.h>
11 #include <linux/random.h>
12 #include <linux/sched.h>
13 #include <linux/usb.h>
14 #include <linux/wait.h>
18 #define LG_G15_TRANSFER_BUF_SIZE 20
20 #define LG_G15_FEATURE_REPORT 0x02
22 #define LG_G510_FEATURE_M_KEYS_LEDS 0x04
23 #define LG_G510_FEATURE_BACKLIGHT_RGB 0x05
24 #define LG_G510_FEATURE_POWER_ON_RGB 0x06
33 enum lg_g15_led_type
{
34 LG_G15_KBD_BRIGHTNESS
,
35 LG_G15_LCD_BRIGHTNESS
,
36 LG_G15_BRIGHTNESS_MAX
,
37 LG_G15_MACRO_PRESET1
= 2,
45 struct led_classdev cdev
;
46 enum led_brightness brightness
;
47 enum lg_g15_led_type led
;
52 /* Must be first for proper dma alignment */
53 u8 transfer_buf
[LG_G15_TRANSFER_BUF_SIZE
];
54 /* Protects the transfer_buf and led brightness */
56 struct work_struct work
;
57 struct input_dev
*input
;
58 struct hid_device
*hdev
;
59 enum lg_g15_model model
;
60 struct lg_g15_led leds
[LG_G15_LED_MAX
];
61 bool game_mode_enabled
;
64 /******** G15 and G15 v2 LED functions ********/
66 static int lg_g15_update_led_brightness(struct lg_g15_data
*g15
)
70 ret
= hid_hw_raw_request(g15
->hdev
, LG_G15_FEATURE_REPORT
,
72 HID_FEATURE_REPORT
, HID_REQ_GET_REPORT
);
74 hid_err(g15
->hdev
, "Error getting LED brightness: %d\n", ret
);
75 return (ret
< 0) ? ret
: -EIO
;
78 g15
->leds
[LG_G15_KBD_BRIGHTNESS
].brightness
= g15
->transfer_buf
[1];
79 g15
->leds
[LG_G15_LCD_BRIGHTNESS
].brightness
= g15
->transfer_buf
[2];
81 g15
->leds
[LG_G15_MACRO_PRESET1
].brightness
=
82 !(g15
->transfer_buf
[3] & 0x01);
83 g15
->leds
[LG_G15_MACRO_PRESET2
].brightness
=
84 !(g15
->transfer_buf
[3] & 0x02);
85 g15
->leds
[LG_G15_MACRO_PRESET3
].brightness
=
86 !(g15
->transfer_buf
[3] & 0x04);
87 g15
->leds
[LG_G15_MACRO_RECORD
].brightness
=
88 !(g15
->transfer_buf
[3] & 0x08);
93 static enum led_brightness
lg_g15_led_get(struct led_classdev
*led_cdev
)
95 struct lg_g15_led
*g15_led
=
96 container_of(led_cdev
, struct lg_g15_led
, cdev
);
97 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
98 enum led_brightness brightness
;
100 mutex_lock(&g15
->mutex
);
101 lg_g15_update_led_brightness(g15
);
102 brightness
= g15
->leds
[g15_led
->led
].brightness
;
103 mutex_unlock(&g15
->mutex
);
108 static int lg_g15_led_set(struct led_classdev
*led_cdev
,
109 enum led_brightness brightness
)
111 struct lg_g15_led
*g15_led
=
112 container_of(led_cdev
, struct lg_g15_led
, cdev
);
113 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
117 /* Ignore LED off on unregister / keyboard unplug */
118 if (led_cdev
->flags
& LED_UNREGISTERING
)
121 mutex_lock(&g15
->mutex
);
123 g15
->transfer_buf
[0] = LG_G15_FEATURE_REPORT
;
124 g15
->transfer_buf
[3] = 0;
126 if (g15_led
->led
< LG_G15_BRIGHTNESS_MAX
) {
127 g15
->transfer_buf
[1] = g15_led
->led
+ 1;
128 g15
->transfer_buf
[2] = brightness
<< (g15_led
->led
* 4);
130 for (i
= LG_G15_MACRO_PRESET1
; i
< LG_G15_LED_MAX
; i
++) {
131 if (i
== g15_led
->led
)
134 val
= g15
->leds
[i
].brightness
;
137 mask
|= 1 << (i
- LG_G15_MACRO_PRESET1
);
140 g15
->transfer_buf
[1] = 0x04;
141 g15
->transfer_buf
[2] = ~mask
;
144 ret
= hid_hw_raw_request(g15
->hdev
, LG_G15_FEATURE_REPORT
,
145 g15
->transfer_buf
, 4,
146 HID_FEATURE_REPORT
, HID_REQ_SET_REPORT
);
149 g15_led
->brightness
= brightness
;
152 hid_err(g15
->hdev
, "Error setting LED brightness: %d\n", ret
);
153 ret
= (ret
< 0) ? ret
: -EIO
;
156 mutex_unlock(&g15
->mutex
);
161 static void lg_g15_leds_changed_work(struct work_struct
*work
)
163 struct lg_g15_data
*g15
= container_of(work
, struct lg_g15_data
, work
);
164 enum led_brightness old_brightness
[LG_G15_BRIGHTNESS_MAX
];
165 enum led_brightness brightness
[LG_G15_BRIGHTNESS_MAX
];
168 mutex_lock(&g15
->mutex
);
169 for (i
= 0; i
< LG_G15_BRIGHTNESS_MAX
; i
++)
170 old_brightness
[i
] = g15
->leds
[i
].brightness
;
172 ret
= lg_g15_update_led_brightness(g15
);
174 for (i
= 0; i
< LG_G15_BRIGHTNESS_MAX
; i
++)
175 brightness
[i
] = g15
->leds
[i
].brightness
;
176 mutex_unlock(&g15
->mutex
);
181 for (i
= 0; i
< LG_G15_BRIGHTNESS_MAX
; i
++) {
182 if (brightness
[i
] == old_brightness
[i
])
185 led_classdev_notify_brightness_hw_changed(&g15
->leds
[i
].cdev
,
190 /******** G510 LED functions ********/
192 static int lg_g510_get_initial_led_brightness(struct lg_g15_data
*g15
, int i
)
196 ret
= hid_hw_raw_request(g15
->hdev
, LG_G510_FEATURE_BACKLIGHT_RGB
+ i
,
197 g15
->transfer_buf
, 4,
198 HID_FEATURE_REPORT
, HID_REQ_GET_REPORT
);
200 hid_err(g15
->hdev
, "Error getting LED brightness: %d\n", ret
);
201 return (ret
< 0) ? ret
: -EIO
;
204 high
= max3(g15
->transfer_buf
[1], g15
->transfer_buf
[2],
205 g15
->transfer_buf
[3]);
209 DIV_ROUND_CLOSEST(g15
->transfer_buf
[1] * 255, high
);
211 DIV_ROUND_CLOSEST(g15
->transfer_buf
[2] * 255, high
);
213 DIV_ROUND_CLOSEST(g15
->transfer_buf
[3] * 255, high
);
214 g15
->leds
[i
].brightness
= high
;
216 g15
->leds
[i
].red
= 255;
217 g15
->leds
[i
].green
= 255;
218 g15
->leds
[i
].blue
= 255;
219 g15
->leds
[i
].brightness
= 0;
225 /* Must be called with g15->mutex locked */
226 static int lg_g510_kbd_led_write(struct lg_g15_data
*g15
,
227 struct lg_g15_led
*g15_led
,
228 enum led_brightness brightness
)
232 g15
->transfer_buf
[0] = 5 + g15_led
->led
;
233 g15
->transfer_buf
[1] =
234 DIV_ROUND_CLOSEST(g15_led
->red
* brightness
, 255);
235 g15
->transfer_buf
[2] =
236 DIV_ROUND_CLOSEST(g15_led
->green
* brightness
, 255);
237 g15
->transfer_buf
[3] =
238 DIV_ROUND_CLOSEST(g15_led
->blue
* brightness
, 255);
240 ret
= hid_hw_raw_request(g15
->hdev
,
241 LG_G510_FEATURE_BACKLIGHT_RGB
+ g15_led
->led
,
242 g15
->transfer_buf
, 4,
243 HID_FEATURE_REPORT
, HID_REQ_SET_REPORT
);
246 g15_led
->brightness
= brightness
;
249 hid_err(g15
->hdev
, "Error setting LED brightness: %d\n", ret
);
250 ret
= (ret
< 0) ? ret
: -EIO
;
256 static int lg_g510_kbd_led_set(struct led_classdev
*led_cdev
,
257 enum led_brightness brightness
)
259 struct lg_g15_led
*g15_led
=
260 container_of(led_cdev
, struct lg_g15_led
, cdev
);
261 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
264 /* Ignore LED off on unregister / keyboard unplug */
265 if (led_cdev
->flags
& LED_UNREGISTERING
)
268 mutex_lock(&g15
->mutex
);
269 ret
= lg_g510_kbd_led_write(g15
, g15_led
, brightness
);
270 mutex_unlock(&g15
->mutex
);
275 static enum led_brightness
lg_g510_kbd_led_get(struct led_classdev
*led_cdev
)
277 struct lg_g15_led
*g15_led
=
278 container_of(led_cdev
, struct lg_g15_led
, cdev
);
280 return g15_led
->brightness
;
283 static ssize_t
color_store(struct device
*dev
, struct device_attribute
*attr
,
284 const char *buf
, size_t count
)
286 struct led_classdev
*led_cdev
= dev_get_drvdata(dev
);
287 struct lg_g15_led
*g15_led
=
288 container_of(led_cdev
, struct lg_g15_led
, cdev
);
289 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
293 if (count
< 7 || (count
== 8 && buf
[7] != '\n') || count
> 8)
299 ret
= kstrtoul(buf
+ 1, 16, &value
);
303 mutex_lock(&g15
->mutex
);
304 g15_led
->red
= (value
& 0xff0000) >> 16;
305 g15_led
->green
= (value
& 0x00ff00) >> 8;
306 g15_led
->blue
= (value
& 0x0000ff);
307 ret
= lg_g510_kbd_led_write(g15
, g15_led
, g15_led
->brightness
);
308 mutex_unlock(&g15
->mutex
);
310 return (ret
< 0) ? ret
: count
;
313 static ssize_t
color_show(struct device
*dev
, struct device_attribute
*attr
,
316 struct led_classdev
*led_cdev
= dev_get_drvdata(dev
);
317 struct lg_g15_led
*g15_led
=
318 container_of(led_cdev
, struct lg_g15_led
, cdev
);
319 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
322 mutex_lock(&g15
->mutex
);
323 ret
= sprintf(buf
, "#%02x%02x%02x\n",
324 g15_led
->red
, g15_led
->green
, g15_led
->blue
);
325 mutex_unlock(&g15
->mutex
);
330 static DEVICE_ATTR_RW(color
);
332 static struct attribute
*lg_g510_kbd_led_attrs
[] = {
333 &dev_attr_color
.attr
,
337 static const struct attribute_group lg_g510_kbd_led_group
= {
338 .attrs
= lg_g510_kbd_led_attrs
,
341 static const struct attribute_group
*lg_g510_kbd_led_groups
[] = {
342 &lg_g510_kbd_led_group
,
346 static void lg_g510_leds_sync_work(struct work_struct
*work
)
348 struct lg_g15_data
*g15
= container_of(work
, struct lg_g15_data
, work
);
350 mutex_lock(&g15
->mutex
);
351 lg_g510_kbd_led_write(g15
, &g15
->leds
[LG_G15_KBD_BRIGHTNESS
],
352 g15
->leds
[LG_G15_KBD_BRIGHTNESS
].brightness
);
353 mutex_unlock(&g15
->mutex
);
356 static int lg_g510_update_mkey_led_brightness(struct lg_g15_data
*g15
)
360 ret
= hid_hw_raw_request(g15
->hdev
, LG_G510_FEATURE_M_KEYS_LEDS
,
361 g15
->transfer_buf
, 2,
362 HID_FEATURE_REPORT
, HID_REQ_GET_REPORT
);
364 hid_err(g15
->hdev
, "Error getting LED brightness: %d\n", ret
);
365 ret
= (ret
< 0) ? ret
: -EIO
;
368 g15
->leds
[LG_G15_MACRO_PRESET1
].brightness
=
369 !!(g15
->transfer_buf
[1] & 0x80);
370 g15
->leds
[LG_G15_MACRO_PRESET2
].brightness
=
371 !!(g15
->transfer_buf
[1] & 0x40);
372 g15
->leds
[LG_G15_MACRO_PRESET3
].brightness
=
373 !!(g15
->transfer_buf
[1] & 0x20);
374 g15
->leds
[LG_G15_MACRO_RECORD
].brightness
=
375 !!(g15
->transfer_buf
[1] & 0x10);
380 static enum led_brightness
lg_g510_mkey_led_get(struct led_classdev
*led_cdev
)
382 struct lg_g15_led
*g15_led
=
383 container_of(led_cdev
, struct lg_g15_led
, cdev
);
384 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
385 enum led_brightness brightness
;
387 mutex_lock(&g15
->mutex
);
388 lg_g510_update_mkey_led_brightness(g15
);
389 brightness
= g15
->leds
[g15_led
->led
].brightness
;
390 mutex_unlock(&g15
->mutex
);
395 static int lg_g510_mkey_led_set(struct led_classdev
*led_cdev
,
396 enum led_brightness brightness
)
398 struct lg_g15_led
*g15_led
=
399 container_of(led_cdev
, struct lg_g15_led
, cdev
);
400 struct lg_g15_data
*g15
= dev_get_drvdata(led_cdev
->dev
->parent
);
404 /* Ignore LED off on unregister / keyboard unplug */
405 if (led_cdev
->flags
& LED_UNREGISTERING
)
408 mutex_lock(&g15
->mutex
);
410 for (i
= LG_G15_MACRO_PRESET1
; i
< LG_G15_LED_MAX
; i
++) {
411 if (i
== g15_led
->led
)
414 val
= g15
->leds
[i
].brightness
;
417 mask
|= 0x80 >> (i
- LG_G15_MACRO_PRESET1
);
420 g15
->transfer_buf
[0] = LG_G510_FEATURE_M_KEYS_LEDS
;
421 g15
->transfer_buf
[1] = mask
;
423 ret
= hid_hw_raw_request(g15
->hdev
, LG_G510_FEATURE_M_KEYS_LEDS
,
424 g15
->transfer_buf
, 2,
425 HID_FEATURE_REPORT
, HID_REQ_SET_REPORT
);
428 g15_led
->brightness
= brightness
;
431 hid_err(g15
->hdev
, "Error setting LED brightness: %d\n", ret
);
432 ret
= (ret
< 0) ? ret
: -EIO
;
435 mutex_unlock(&g15
->mutex
);
440 /******** Generic LED functions ********/
441 static int lg_g15_get_initial_led_brightness(struct lg_g15_data
*g15
)
445 switch (g15
->model
) {
448 return lg_g15_update_led_brightness(g15
);
450 case LG_G510_USB_AUDIO
:
451 ret
= lg_g510_get_initial_led_brightness(g15
, 0);
455 ret
= lg_g510_get_initial_led_brightness(g15
, 1);
459 return lg_g510_update_mkey_led_brightness(g15
);
461 return -EINVAL
; /* Never reached */
464 /******** Input functions ********/
466 /* On the G15 Mark I Logitech has been quite creative with which bit is what */
467 static int lg_g15_event(struct lg_g15_data
*g15
, u8
*data
, int size
)
472 for (i
= 0; i
< 6; i
++) {
473 val
= data
[i
+ 1] & (1 << i
);
474 input_report_key(g15
->input
, KEY_MACRO1
+ i
, val
);
477 for (i
= 0; i
< 6; i
++) {
478 val
= data
[i
+ 2] & (1 << i
);
479 input_report_key(g15
->input
, KEY_MACRO7
+ i
, val
);
482 for (i
= 0; i
< 5; i
++) {
483 val
= data
[i
+ 1] & (4 << i
);
484 input_report_key(g15
->input
, KEY_MACRO13
+ i
, val
);
487 input_report_key(g15
->input
, KEY_MACRO18
, data
[8] & 0x40);
490 for (i
= 0; i
< 3; i
++) {
491 val
= data
[i
+ 6] & (1 << i
);
492 input_report_key(g15
->input
, KEY_MACRO_PRESET1
+ i
, val
);
495 input_report_key(g15
->input
, KEY_MACRO_RECORD_START
, data
[7] & 0x40);
497 /* Most left (round) button below the LCD */
498 input_report_key(g15
->input
, KEY_KBD_LCD_MENU1
, data
[8] & 0x80);
499 /* 4 other buttons below the LCD */
500 for (i
= 0; i
< 4; i
++) {
501 val
= data
[i
+ 2] & 0x80;
502 input_report_key(g15
->input
, KEY_KBD_LCD_MENU2
+ i
, val
);
505 /* Backlight cycle button pressed? */
507 schedule_work(&g15
->work
);
509 input_sync(g15
->input
);
513 static int lg_g15_v2_event(struct lg_g15_data
*g15
, u8
*data
, int size
)
518 for (i
= 0; i
< 6; i
++) {
519 val
= data
[1] & (1 << i
);
520 input_report_key(g15
->input
, KEY_MACRO1
+ i
, val
);
524 input_report_key(g15
->input
, KEY_MACRO_PRESET1
, data
[1] & 0x40);
525 input_report_key(g15
->input
, KEY_MACRO_PRESET2
, data
[1] & 0x80);
526 input_report_key(g15
->input
, KEY_MACRO_PRESET3
, data
[2] & 0x20);
527 input_report_key(g15
->input
, KEY_MACRO_RECORD_START
, data
[2] & 0x40);
529 /* Round button to the left of the LCD */
530 input_report_key(g15
->input
, KEY_KBD_LCD_MENU1
, data
[2] & 0x80);
531 /* 4 buttons below the LCD */
532 for (i
= 0; i
< 4; i
++) {
533 val
= data
[2] & (2 << i
);
534 input_report_key(g15
->input
, KEY_KBD_LCD_MENU2
+ i
, val
);
537 /* Backlight cycle button pressed? */
539 schedule_work(&g15
->work
);
541 input_sync(g15
->input
);
545 static int lg_g510_event(struct lg_g15_data
*g15
, u8
*data
, int size
)
547 bool game_mode_enabled
;
551 for (i
= 0; i
< 18; i
++) {
552 val
= data
[i
/ 8 + 1] & (1 << (i
% 8));
553 input_report_key(g15
->input
, KEY_MACRO1
+ i
, val
);
556 /* Game mode on/off slider */
557 game_mode_enabled
= data
[3] & 0x04;
558 if (game_mode_enabled
!= g15
->game_mode_enabled
) {
559 if (game_mode_enabled
)
560 hid_info(g15
->hdev
, "Game Mode enabled, Windows (super) key is disabled\n");
562 hid_info(g15
->hdev
, "Game Mode disabled\n");
563 g15
->game_mode_enabled
= game_mode_enabled
;
567 for (i
= 0; i
< 3; i
++) {
568 val
= data
[3] & (0x10 << i
);
569 input_report_key(g15
->input
, KEY_MACRO_PRESET1
+ i
, val
);
572 input_report_key(g15
->input
, KEY_MACRO_RECORD_START
, data
[3] & 0x80);
575 for (i
= 0; i
< 5; i
++) {
576 val
= data
[4] & (1 << i
);
577 input_report_key(g15
->input
, KEY_KBD_LCD_MENU1
+ i
, val
);
581 input_report_key(g15
->input
, KEY_MUTE
, data
[4] & 0x20);
582 /* Microphone Mute */
583 input_report_key(g15
->input
, KEY_F20
, data
[4] & 0x40);
585 input_sync(g15
->input
);
589 static int lg_g510_leds_event(struct lg_g15_data
*g15
, u8
*data
, int size
)
591 bool backlight_disabled
;
594 * The G510 ignores backlight updates when the backlight is turned off
595 * through the light toggle button on the keyboard, to work around this
596 * we queue a workitem to sync values when the backlight is turned on.
598 backlight_disabled
= data
[1] & 0x04;
599 if (!backlight_disabled
)
600 schedule_work(&g15
->work
);
605 static int lg_g15_raw_event(struct hid_device
*hdev
, struct hid_report
*report
,
608 struct lg_g15_data
*g15
= hid_get_drvdata(hdev
);
613 switch (g15
->model
) {
615 if (data
[0] == 0x02 && size
== 9)
616 return lg_g15_event(g15
, data
, size
);
619 if (data
[0] == 0x02 && size
== 5)
620 return lg_g15_v2_event(g15
, data
, size
);
623 case LG_G510_USB_AUDIO
:
624 if (data
[0] == 0x03 && size
== 5)
625 return lg_g510_event(g15
, data
, size
);
626 if (data
[0] == 0x04 && size
== 2)
627 return lg_g510_leds_event(g15
, data
, size
);
634 static int lg_g15_input_open(struct input_dev
*dev
)
636 struct hid_device
*hdev
= input_get_drvdata(dev
);
638 return hid_hw_open(hdev
);
641 static void lg_g15_input_close(struct input_dev
*dev
)
643 struct hid_device
*hdev
= input_get_drvdata(dev
);
648 static int lg_g15_register_led(struct lg_g15_data
*g15
, int i
)
650 const char * const led_names
[] = {
651 "g15::kbd_backlight",
652 "g15::lcd_backlight",
653 "g15::macro_preset1",
654 "g15::macro_preset2",
655 "g15::macro_preset3",
659 g15
->leds
[i
].led
= i
;
660 g15
->leds
[i
].cdev
.name
= led_names
[i
];
662 switch (g15
->model
) {
665 g15
->leds
[i
].cdev
.brightness_set_blocking
= lg_g15_led_set
;
666 g15
->leds
[i
].cdev
.brightness_get
= lg_g15_led_get
;
667 if (i
< LG_G15_BRIGHTNESS_MAX
) {
668 g15
->leds
[i
].cdev
.flags
= LED_BRIGHT_HW_CHANGED
;
669 g15
->leds
[i
].cdev
.max_brightness
= 2;
671 g15
->leds
[i
].cdev
.max_brightness
= 1;
675 case LG_G510_USB_AUDIO
:
677 case LG_G15_LCD_BRIGHTNESS
:
679 * The G510 does not have a separate LCD brightness,
680 * but it does have a separate power-on (reset) value.
682 g15
->leds
[i
].cdev
.name
= "g15::power_on_backlight_val";
684 case LG_G15_KBD_BRIGHTNESS
:
685 g15
->leds
[i
].cdev
.brightness_set_blocking
=
687 g15
->leds
[i
].cdev
.brightness_get
=
689 g15
->leds
[i
].cdev
.max_brightness
= 255;
690 g15
->leds
[i
].cdev
.groups
= lg_g510_kbd_led_groups
;
693 g15
->leds
[i
].cdev
.brightness_set_blocking
=
694 lg_g510_mkey_led_set
;
695 g15
->leds
[i
].cdev
.brightness_get
=
696 lg_g510_mkey_led_get
;
697 g15
->leds
[i
].cdev
.max_brightness
= 1;
702 return devm_led_classdev_register(&g15
->hdev
->dev
, &g15
->leds
[i
].cdev
);
705 static int lg_g15_probe(struct hid_device
*hdev
, const struct hid_device_id
*id
)
707 u8 gkeys_settings_output_report
= 0;
708 u8 gkeys_settings_feature_report
= 0;
709 struct hid_report_enum
*rep_enum
;
710 unsigned int connect_mask
= 0;
711 bool has_ff000000
= false;
712 struct lg_g15_data
*g15
;
713 struct input_dev
*input
;
714 struct hid_report
*rep
;
715 int ret
, i
, gkeys
= 0;
717 hdev
->quirks
|= HID_QUIRK_INPUT_PER_APP
;
719 ret
= hid_parse(hdev
);
724 * Some models have multiple interfaces, we want the interface with
725 * with the f000.0000 application input report.
727 rep_enum
= &hdev
->report_enum
[HID_INPUT_REPORT
];
728 list_for_each_entry(rep
, &rep_enum
->report_list
, list
) {
729 if (rep
->application
== 0xff000000)
733 return hid_hw_start(hdev
, HID_CONNECT_DEFAULT
);
735 g15
= devm_kzalloc(&hdev
->dev
, sizeof(*g15
), GFP_KERNEL
);
739 mutex_init(&g15
->mutex
);
741 input
= devm_input_allocate_device(&hdev
->dev
);
746 g15
->model
= id
->driver_data
;
747 hid_set_drvdata(hdev
, (void *)g15
);
749 switch (g15
->model
) {
751 INIT_WORK(&g15
->work
, lg_g15_leds_changed_work
);
753 * The G15 and G15 v2 use a separate usb-device (on a builtin
754 * hub) which emulates a keyboard for the F1 - F12 emulation
755 * on the G-keys, which we disable, rendering the emulated kbd
756 * non-functional, so we do not let hid-input connect.
758 connect_mask
= HID_CONNECT_HIDRAW
;
759 gkeys_settings_output_report
= 0x02;
763 INIT_WORK(&g15
->work
, lg_g15_leds_changed_work
);
764 connect_mask
= HID_CONNECT_HIDRAW
;
765 gkeys_settings_output_report
= 0x02;
769 case LG_G510_USB_AUDIO
:
770 INIT_WORK(&g15
->work
, lg_g510_leds_sync_work
);
771 connect_mask
= HID_CONNECT_HIDINPUT
| HID_CONNECT_HIDRAW
;
772 gkeys_settings_feature_report
= 0x01;
777 ret
= hid_hw_start(hdev
, connect_mask
);
781 /* Tell the keyboard to stop sending F1-F12 + 1-6 for G1 - G18 */
782 if (gkeys_settings_output_report
) {
783 g15
->transfer_buf
[0] = gkeys_settings_output_report
;
784 memset(g15
->transfer_buf
+ 1, 0, gkeys
);
786 * The kbd ignores our output report if we do not queue
787 * an URB on the USB input endpoint first...
789 ret
= hid_hw_open(hdev
);
792 ret
= hid_hw_output_report(hdev
, g15
->transfer_buf
, gkeys
+ 1);
796 if (gkeys_settings_feature_report
) {
797 g15
->transfer_buf
[0] = gkeys_settings_feature_report
;
798 memset(g15
->transfer_buf
+ 1, 0, gkeys
);
799 ret
= hid_hw_raw_request(g15
->hdev
,
800 gkeys_settings_feature_report
,
801 g15
->transfer_buf
, gkeys
+ 1,
802 HID_FEATURE_REPORT
, HID_REQ_SET_REPORT
);
806 hid_err(hdev
, "Error disabling keyboard emulation for the G-keys\n");
810 /* Get initial brightness levels */
811 ret
= lg_g15_get_initial_led_brightness(g15
);
815 /* Setup and register input device */
816 input
->name
= "Logitech Gaming Keyboard Gaming Keys";
817 input
->phys
= hdev
->phys
;
818 input
->uniq
= hdev
->uniq
;
819 input
->id
.bustype
= hdev
->bus
;
820 input
->id
.vendor
= hdev
->vendor
;
821 input
->id
.product
= hdev
->product
;
822 input
->id
.version
= hdev
->version
;
823 input
->dev
.parent
= &hdev
->dev
;
824 input
->open
= lg_g15_input_open
;
825 input
->close
= lg_g15_input_close
;
828 for (i
= 0; i
< gkeys
; i
++)
829 input_set_capability(input
, EV_KEY
, KEY_MACRO1
+ i
);
831 /* M1 - M3 and MR keys */
832 for (i
= 0; i
< 3; i
++)
833 input_set_capability(input
, EV_KEY
, KEY_MACRO_PRESET1
+ i
);
834 input_set_capability(input
, EV_KEY
, KEY_MACRO_RECORD_START
);
836 /* Keys below the LCD, intended for controlling a menu on the LCD */
837 for (i
= 0; i
< 5; i
++)
838 input_set_capability(input
, EV_KEY
, KEY_KBD_LCD_MENU1
+ i
);
841 * On the G510 only report headphone and mic mute keys when *not* using
842 * the builtin USB audio device. When the builtin audio is used these
843 * keys directly toggle mute (and the LEDs) on/off.
845 if (g15
->model
== LG_G510
) {
846 input_set_capability(input
, EV_KEY
, KEY_MUTE
);
847 /* Userspace expects F20 for micmute */
848 input_set_capability(input
, EV_KEY
, KEY_F20
);
852 input_set_drvdata(input
, hdev
);
854 ret
= input_register_device(input
);
858 /* Register LED devices */
859 for (i
= 0; i
< LG_G15_LED_MAX
; i
++) {
860 ret
= lg_g15_register_led(g15
, i
);
872 static const struct hid_device_id lg_g15_devices
[] = {
873 { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH
,
874 USB_DEVICE_ID_LOGITECH_G15_LCD
),
875 .driver_data
= LG_G15
},
876 { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH
,
877 USB_DEVICE_ID_LOGITECH_G15_V2_LCD
),
878 .driver_data
= LG_G15_V2
},
879 /* G510 without a headset plugged in */
880 { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH
,
881 USB_DEVICE_ID_LOGITECH_G510
),
882 .driver_data
= LG_G510
},
883 /* G510 with headset plugged in / with extra USB audio interface */
884 { HID_USB_DEVICE(USB_VENDOR_ID_LOGITECH
,
885 USB_DEVICE_ID_LOGITECH_G510_USB_AUDIO
),
886 .driver_data
= LG_G510_USB_AUDIO
},
889 MODULE_DEVICE_TABLE(hid
, lg_g15_devices
);
891 static struct hid_driver lg_g15_driver
= {
893 .id_table
= lg_g15_devices
,
894 .raw_event
= lg_g15_raw_event
,
895 .probe
= lg_g15_probe
,
897 module_hid_driver(lg_g15_driver
);
899 MODULE_LICENSE("GPL");