2 * ThingM blink(1) USB RGB LED driver
4 * Copyright 2013-2014 Savoir-faire Linux Inc.
5 * Vivien Didelot <vivien.didelot@savoirfairelinux.com>
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License as
9 * published by the Free Software Foundation, version 2.
12 #include <linux/hid.h>
13 #include <linux/hidraw.h>
14 #include <linux/leds.h>
15 #include <linux/module.h>
16 #include <linux/mutex.h>
17 #include <linux/workqueue.h>
24 /* Firmware major number of supported devices */
25 #define THINGM_MAJOR_MK1 '1'
26 #define THINGM_MAJOR_MK2 '2'
28 struct thingm_fwinfo
{
34 static const struct thingm_fwinfo thingm_fwinfo
[] = {
36 .major
= THINGM_MAJOR_MK1
,
40 .major
= THINGM_MAJOR_MK2
,
46 /* A red, green or blue channel, part of an RGB chip */
48 struct thingm_rgb
*rgb
;
49 struct led_classdev ldev
;
53 /* Basically a WS2812 5050 RGB LED chip */
55 struct thingm_device
*tdev
;
56 struct thingm_led red
;
57 struct thingm_led green
;
58 struct thingm_led blue
;
59 struct work_struct work
;
63 struct thingm_device
{
64 struct hid_device
*hdev
;
69 const struct thingm_fwinfo
*fwinfo
;
71 struct thingm_rgb
*rgb
;
74 static int thingm_send(struct thingm_device
*tdev
, u8 buf
[REPORT_SIZE
])
78 hid_dbg(tdev
->hdev
, "-> %d %c %02hhx %02hhx %02hhx %02hhx %02hhx %02hhx %02hhx\n",
79 buf
[0], buf
[1], buf
[2], buf
[3], buf
[4],
80 buf
[5], buf
[6], buf
[7], buf
[8]);
82 ret
= hid_hw_raw_request(tdev
->hdev
, buf
[0], buf
, REPORT_SIZE
,
83 HID_FEATURE_REPORT
, HID_REQ_SET_REPORT
);
85 return ret
< 0 ? ret
: 0;
88 static int thingm_recv(struct thingm_device
*tdev
, u8 buf
[REPORT_SIZE
])
92 ret
= hid_hw_raw_request(tdev
->hdev
, buf
[0], buf
, REPORT_SIZE
,
93 HID_FEATURE_REPORT
, HID_REQ_GET_REPORT
);
97 hid_dbg(tdev
->hdev
, "<- %d %c %02hhx %02hhx %02hhx %02hhx %02hhx %02hhx %02hhx\n",
98 buf
[0], buf
[1], buf
[2], buf
[3], buf
[4],
99 buf
[5], buf
[6], buf
[7], buf
[8]);
104 static int thingm_version(struct thingm_device
*tdev
)
106 u8 buf
[REPORT_SIZE
] = { REPORT_ID
, 'v', 0, 0, 0, 0, 0, 0, 0 };
109 err
= thingm_send(tdev
, buf
);
113 err
= thingm_recv(tdev
, buf
);
117 tdev
->version
.major
= buf
[3];
118 tdev
->version
.minor
= buf
[4];
123 static int thingm_write_color(struct thingm_rgb
*rgb
)
125 u8 buf
[REPORT_SIZE
] = { REPORT_ID
, 'c', 0, 0, 0, 0, 0, rgb
->num
, 0 };
127 buf
[2] = rgb
->red
.ldev
.brightness
;
128 buf
[3] = rgb
->green
.ldev
.brightness
;
129 buf
[4] = rgb
->blue
.ldev
.brightness
;
131 return thingm_send(rgb
->tdev
, buf
);
134 static void thingm_work(struct work_struct
*work
)
136 struct thingm_rgb
*rgb
= container_of(work
, struct thingm_rgb
, work
);
138 mutex_lock(&rgb
->tdev
->lock
);
140 if (thingm_write_color(rgb
))
141 hid_err(rgb
->tdev
->hdev
, "failed to write color\n");
143 mutex_unlock(&rgb
->tdev
->lock
);
146 static void thingm_led_set(struct led_classdev
*ldev
,
147 enum led_brightness brightness
)
149 struct thingm_led
*led
= container_of(ldev
, struct thingm_led
, ldev
);
151 /* the ledclass has already stored the brightness value */
152 schedule_work(&led
->rgb
->work
);
155 static int thingm_init_rgb(struct thingm_rgb
*rgb
)
157 const int minor
= ((struct hidraw
*) rgb
->tdev
->hdev
->hidraw
)->minor
;
160 /* Register the red diode */
161 snprintf(rgb
->red
.name
, sizeof(rgb
->red
.name
),
162 "thingm%d:red:led%d", minor
, rgb
->num
);
163 rgb
->red
.ldev
.name
= rgb
->red
.name
;
164 rgb
->red
.ldev
.max_brightness
= 255;
165 rgb
->red
.ldev
.brightness_set
= thingm_led_set
;
168 err
= led_classdev_register(&rgb
->tdev
->hdev
->dev
, &rgb
->red
.ldev
);
172 /* Register the green diode */
173 snprintf(rgb
->green
.name
, sizeof(rgb
->green
.name
),
174 "thingm%d:green:led%d", minor
, rgb
->num
);
175 rgb
->green
.ldev
.name
= rgb
->green
.name
;
176 rgb
->green
.ldev
.max_brightness
= 255;
177 rgb
->green
.ldev
.brightness_set
= thingm_led_set
;
178 rgb
->green
.rgb
= rgb
;
180 err
= led_classdev_register(&rgb
->tdev
->hdev
->dev
, &rgb
->green
.ldev
);
184 /* Register the blue diode */
185 snprintf(rgb
->blue
.name
, sizeof(rgb
->blue
.name
),
186 "thingm%d:blue:led%d", minor
, rgb
->num
);
187 rgb
->blue
.ldev
.name
= rgb
->blue
.name
;
188 rgb
->blue
.ldev
.max_brightness
= 255;
189 rgb
->blue
.ldev
.brightness_set
= thingm_led_set
;
192 err
= led_classdev_register(&rgb
->tdev
->hdev
->dev
, &rgb
->blue
.ldev
);
194 goto unregister_green
;
196 INIT_WORK(&rgb
->work
, thingm_work
);
201 led_classdev_unregister(&rgb
->green
.ldev
);
204 led_classdev_unregister(&rgb
->red
.ldev
);
209 static void thingm_remove_rgb(struct thingm_rgb
*rgb
)
211 flush_work(&rgb
->work
);
212 led_classdev_unregister(&rgb
->red
.ldev
);
213 led_classdev_unregister(&rgb
->green
.ldev
);
214 led_classdev_unregister(&rgb
->blue
.ldev
);
217 static int thingm_probe(struct hid_device
*hdev
, const struct hid_device_id
*id
)
219 struct thingm_device
*tdev
;
222 tdev
= devm_kzalloc(&hdev
->dev
, sizeof(struct thingm_device
),
228 hid_set_drvdata(hdev
, tdev
);
230 err
= hid_parse(hdev
);
234 err
= hid_hw_start(hdev
, HID_CONNECT_HIDRAW
);
238 mutex_init(&tdev
->lock
);
240 err
= thingm_version(tdev
);
244 hid_dbg(hdev
, "firmware version: %c.%c\n",
245 tdev
->version
.major
, tdev
->version
.minor
);
247 for (i
= 0; i
< ARRAY_SIZE(thingm_fwinfo
) && !tdev
->fwinfo
; ++i
)
248 if (thingm_fwinfo
[i
].major
== tdev
->version
.major
)
249 tdev
->fwinfo
= &thingm_fwinfo
[i
];
252 hid_err(hdev
, "unsupported firmware %c\n", tdev
->version
.major
);
256 tdev
->rgb
= devm_kzalloc(&hdev
->dev
,
257 sizeof(struct thingm_rgb
) * tdev
->fwinfo
->numrgb
,
264 for (i
= 0; i
< tdev
->fwinfo
->numrgb
; ++i
) {
265 struct thingm_rgb
*rgb
= tdev
->rgb
+ i
;
268 rgb
->num
= tdev
->fwinfo
->first
+ i
;
269 err
= thingm_init_rgb(rgb
);
272 thingm_remove_rgb(tdev
->rgb
+ i
);
284 static void thingm_remove(struct hid_device
*hdev
)
286 struct thingm_device
*tdev
= hid_get_drvdata(hdev
);
289 for (i
= 0; i
< tdev
->fwinfo
->numrgb
; ++i
)
290 thingm_remove_rgb(tdev
->rgb
+ i
);
295 static const struct hid_device_id thingm_table
[] = {
296 { HID_USB_DEVICE(USB_VENDOR_ID_THINGM
, USB_DEVICE_ID_BLINK1
) },
299 MODULE_DEVICE_TABLE(hid
, thingm_table
);
301 static struct hid_driver thingm_driver
= {
303 .probe
= thingm_probe
,
304 .remove
= thingm_remove
,
305 .id_table
= thingm_table
,
308 module_hid_driver(thingm_driver
);
310 MODULE_LICENSE("GPL");
311 MODULE_AUTHOR("Vivien Didelot <vivien.didelot@savoirfairelinux.com>");
312 MODULE_DESCRIPTION("ThingM blink(1) USB RGB LED driver");