1 // SPDX-License-Identifier: GPL-2.0-or-later
4 * Infrared Toy and IR Droid RC core driver
6 * Copyright (C) 2020 Sean Young <sean@mess.org>
8 * http://dangerousprototypes.com/docs/USB_IR_Toy:_Sampling_mode
10 * This driver is based on the lirc driver which can be found here:
11 * https://sourceforge.net/p/lirc/git/ci/master/tree/plugins/irtoy.c
12 * Copyright (C) 2011 Peter Kooiman <pkooiman@gmail.com>
15 #include <asm/unaligned.h>
16 #include <linux/completion.h>
17 #include <linux/kernel.h>
18 #include <linux/module.h>
19 #include <linux/usb.h>
20 #include <linux/slab.h>
21 #include <linux/usb/input.h>
23 #include <media/rc-core.h>
25 static const u8 COMMAND_VERSION[] = { 'v' };
26 // End transmit and repeat reset command so we exit sump mode
27 static const u8 COMMAND_RESET[] = { 0xff, 0xff, 0, 0, 0, 0, 0 };
28 static const u8 COMMAND_SMODE_ENTER[] = { 's' };
29 static const u8 COMMAND_TXSTART[] = { 0x26, 0x24, 0x25, 0x03 };
31 #define REPLY_XMITCOUNT 't'
32 #define REPLY_XMITSUCCESS 'C'
33 #define REPLY_VERSION 'V'
34 #define REPLY_SAMPLEMODEPROTO 'S'
40 #define LEN_SAMPLEMODEPROTO 3
42 #define MIN_FW_VERSION 20
44 #define MAX_TIMEOUT_US (UNIT_US * U16_MAX)
50 STATE_COMMAND_NO_RESP,
57 struct usb_device *usbdev;
60 struct urb *urb_in, *urb_out;
64 struct completion command_done;
80 static void irtoy_response(struct irtoy *irtoy, u32 len)
82 switch (irtoy->state) {
84 if (len == LEN_VERSION && irtoy->in[0] == REPLY_VERSION) {
87 irtoy->in[LEN_VERSION] = 0;
89 if (kstrtouint(irtoy->in + 1, 10, &version)) {
90 dev_err(irtoy->dev, "invalid version %*phN. Please make sure you are using firmware v20 or higher",
91 LEN_VERSION, irtoy->in);
95 dev_dbg(irtoy->dev, "version %s\n", irtoy->in);
97 irtoy->hw_version = version / 100;
98 irtoy->sw_version = version % 100;
100 irtoy->state = STATE_IRDATA;
101 complete(&irtoy->command_done);
102 } else if (len == LEN_SAMPLEMODEPROTO &&
103 irtoy->in[0] == REPLY_SAMPLEMODEPROTO) {
106 irtoy->in[LEN_SAMPLEMODEPROTO] = 0;
108 if (kstrtouint(irtoy->in + 1, 10, &version)) {
109 dev_err(irtoy->dev, "invalid sample mode response %*phN",
110 LEN_SAMPLEMODEPROTO, irtoy->in);
114 dev_dbg(irtoy->dev, "protocol %s\n", irtoy->in);
116 irtoy->proto_version = version;
118 irtoy->state = STATE_IRDATA;
119 complete(&irtoy->command_done);
121 dev_err(irtoy->dev, "unexpected response to command: %*phN\n",
125 case STATE_COMMAND_NO_RESP:
127 struct ir_raw_event rawir = { .pulse = irtoy->pulse };
128 __be16 *in = (__be16 *)irtoy->in;
131 for (i = 0; i < len / sizeof(__be16); i++) {
132 u16 v = be16_to_cpu(in[i]);
137 rawir.duration = v * UNIT_US;
138 ir_raw_event_store_with_timeout(irtoy->rc,
142 rawir.pulse = !rawir.pulse;
145 irtoy->pulse = rawir.pulse;
147 ir_raw_event_handle(irtoy->rc);
151 if (irtoy->tx_len == 0) {
152 if (len == LEN_XMITRES &&
153 irtoy->in[0] == REPLY_XMITCOUNT) {
154 u16 emitted = get_unaligned_be16(irtoy->in + 1);
156 dev_dbg(irtoy->dev, "emitted:%u\n", emitted);
158 irtoy->emitted = emitted;
159 } else if (len == 1 &&
160 irtoy->in[0] == REPLY_XMITSUCCESS) {
161 irtoy->state = STATE_IRDATA;
162 complete(&irtoy->command_done);
165 // send next part of tx buffer
166 uint space = irtoy->in[0];
170 if (len != 1 || space > MAX_PACKET || space == 0) {
171 dev_dbg(irtoy->dev, "packet length expected: %*phN\n",
176 buf_len = min(space, irtoy->tx_len);
178 dev_dbg(irtoy->dev, "remaining:%u sending:%u\n",
179 irtoy->tx_len, buf_len);
181 memcpy(irtoy->out, irtoy->tx_buf, buf_len);
182 irtoy->urb_out->transfer_buffer_length = buf_len;
183 err = usb_submit_urb(irtoy->urb_out, GFP_ATOMIC);
185 dev_err(irtoy->dev, "fail to submit tx buf urb: %d\n",
187 irtoy->state = STATE_IRDATA;
188 complete(&irtoy->command_done);
192 irtoy->tx_buf += buf_len;
193 irtoy->tx_len -= buf_len;
199 static void irtoy_out_callback(struct urb *urb)
201 struct irtoy *irtoy = urb->context;
203 if (urb->status == 0) {
204 if (irtoy->state == STATE_COMMAND_NO_RESP)
205 complete(&irtoy->command_done);
207 dev_warn(irtoy->dev, "out urb status: %d\n", urb->status);
211 static void irtoy_in_callback(struct urb *urb)
213 struct irtoy *irtoy = urb->context;
216 switch (urb->status) {
218 irtoy_response(irtoy, urb->actual_length);
228 dev_dbg(irtoy->dev, "in urb status: %d\n", urb->status);
231 ret = usb_submit_urb(urb, GFP_ATOMIC);
232 if (ret && ret != -ENODEV)
233 dev_warn(irtoy->dev, "failed to resubmit urb: %d\n", ret);
236 static int irtoy_command(struct irtoy *irtoy, const u8 *cmd, int cmd_len,
241 init_completion(&irtoy->command_done);
243 irtoy->state = state;
245 memcpy(irtoy->out, cmd, cmd_len);
246 irtoy->urb_out->transfer_buffer_length = cmd_len;
248 err = usb_submit_urb(irtoy->urb_out, GFP_KERNEL);
252 if (!wait_for_completion_timeout(&irtoy->command_done,
253 msecs_to_jiffies(TIMEOUT))) {
254 usb_kill_urb(irtoy->urb_out);
261 static int irtoy_setup(struct irtoy *irtoy)
265 err = irtoy_command(irtoy, COMMAND_RESET, sizeof(COMMAND_RESET),
266 STATE_COMMAND_NO_RESP);
268 dev_err(irtoy->dev, "could not write reset command: %d\n",
273 usleep_range(50, 50);
276 err = irtoy_command(irtoy, COMMAND_VERSION, sizeof(COMMAND_VERSION),
279 dev_err(irtoy->dev, "could not write version command: %d\n",
285 err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
286 sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
288 dev_err(irtoy->dev, "could not write sample command: %d\n",
295 * When sending IR, it is imperative that we send the IR data as quickly
296 * as possible to the device, so it does not run out of IR data and
297 * introduce gaps. Allocate the buffer here, and then feed the data from
298 * the urb callback handler.
300 static int irtoy_tx(struct rc_dev *rc, uint *txbuf, uint count)
302 struct irtoy *irtoy = rc->priv;
303 unsigned int i, size;
307 size = sizeof(u16) * (count + 1);
308 buf = kmalloc(size, GFP_KERNEL);
312 for (i = 0; i < count; i++) {
313 u16 v = DIV_ROUND_CLOSEST(txbuf[i], UNIT_US);
317 buf[i] = cpu_to_be16(v);
320 buf[count] = cpu_to_be16(0xffff);
323 irtoy->tx_len = size;
326 err = irtoy_command(irtoy, COMMAND_TXSTART, sizeof(COMMAND_TXSTART),
331 dev_err(irtoy->dev, "failed to send tx start command: %d\n",
333 // not sure what state the device is in, reset it
338 if (size != irtoy->emitted) {
339 dev_err(irtoy->dev, "expected %u emitted, got %u\n", size,
341 // not sure what state the device is in, reset it
349 static int irtoy_tx_carrier(struct rc_dev *rc, uint32_t carrier)
351 struct irtoy *irtoy = rc->priv;
359 buf[1] = DIV_ROUND_CLOSEST(48000000, 16 * carrier) - 1;
362 err = irtoy_command(irtoy, buf, sizeof(buf), STATE_COMMAND_NO_RESP);
364 dev_err(irtoy->dev, "could not write carrier command: %d\n",
370 static int irtoy_probe(struct usb_interface *intf,
371 const struct usb_device_id *id)
373 struct usb_host_interface *idesc = intf->cur_altsetting;
374 struct usb_device *usbdev = interface_to_usbdev(intf);
375 struct usb_endpoint_descriptor *ep_in = NULL;
376 struct usb_endpoint_descriptor *ep_out = NULL;
377 struct usb_endpoint_descriptor *ep = NULL;
381 int i, pipe, err = -ENOMEM;
383 for (i = 0; i < idesc->desc.bNumEndpoints; i++) {
384 ep = &idesc->endpoint[i].desc;
386 if (!ep_in && usb_endpoint_is_bulk_in(ep) &&
387 usb_endpoint_maxp(ep) == MAX_PACKET)
390 if (!ep_out && usb_endpoint_is_bulk_out(ep) &&
391 usb_endpoint_maxp(ep) == MAX_PACKET)
395 if (!ep_in || !ep_out) {
396 dev_err(&intf->dev, "required endpoints not found\n");
400 irtoy = kzalloc(sizeof(*irtoy), GFP_KERNEL);
404 irtoy->in = kmalloc(MAX_PACKET, GFP_KERNEL);
408 irtoy->out = kmalloc(MAX_PACKET, GFP_KERNEL);
412 rc = rc_allocate_device(RC_DRIVER_IR_RAW);
416 urb = usb_alloc_urb(0, GFP_KERNEL);
420 pipe = usb_rcvbulkpipe(usbdev, ep_in->bEndpointAddress);
421 usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->in, MAX_PACKET,
422 irtoy_in_callback, irtoy);
425 urb = usb_alloc_urb(0, GFP_KERNEL);
429 pipe = usb_sndbulkpipe(usbdev, ep_out->bEndpointAddress);
430 usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->out, MAX_PACKET,
431 irtoy_out_callback, irtoy);
433 irtoy->dev = &intf->dev;
434 irtoy->usbdev = usbdev;
436 irtoy->urb_out = urb;
439 err = usb_submit_urb(irtoy->urb_in, GFP_KERNEL);
441 dev_err(irtoy->dev, "fail to submit in urb: %d\n", err);
445 err = irtoy_setup(irtoy);
449 dev_info(irtoy->dev, "version: hardware %u, firmware %u.%u, protocol %u",
450 irtoy->hw_version, irtoy->sw_version / 10,
451 irtoy->sw_version % 10, irtoy->proto_version);
453 if (irtoy->sw_version < MIN_FW_VERSION) {
454 dev_err(irtoy->dev, "need firmware V%02u or higher",
460 usb_make_path(usbdev, irtoy->phys, sizeof(irtoy->phys));
462 rc->device_name = "Infrared Toy";
463 rc->driver_name = KBUILD_MODNAME;
464 rc->input_phys = irtoy->phys;
465 usb_to_input_id(usbdev, &rc->input_id);
466 rc->dev.parent = &intf->dev;
468 rc->tx_ir = irtoy_tx;
469 rc->s_tx_carrier = irtoy_tx_carrier;
470 rc->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER;
471 rc->map_name = RC_MAP_RC6_MCE;
472 rc->rx_resolution = UNIT_US;
473 rc->timeout = IR_DEFAULT_TIMEOUT;
476 * end of transmission is detected by absence of a usb packet
477 * with more pulse/spaces. However, each usb packet sent can
478 * contain 32 pulse/spaces, which can be quite lengthy, so there
479 * can be a delay between usb packets. For example with nec there is a
480 * 17ms gap between packets.
482 * So, make timeout a largish minimum which works with most protocols.
484 rc->min_timeout = MS_TO_US(40);
485 rc->max_timeout = MAX_TIMEOUT_US;
487 err = rc_register_device(rc);
491 usb_set_intfdata(intf, irtoy);
496 usb_kill_urb(irtoy->urb_out);
497 usb_free_urb(irtoy->urb_out);
498 usb_kill_urb(irtoy->urb_in);
499 usb_free_urb(irtoy->urb_in);
508 static void irtoy_disconnect(struct usb_interface *intf)
510 struct irtoy *ir = usb_get_intfdata(intf);
512 rc_unregister_device(ir->rc);
513 usb_set_intfdata(intf, NULL);
514 usb_kill_urb(ir->urb_out);
515 usb_free_urb(ir->urb_out);
516 usb_kill_urb(ir->urb_in);
517 usb_free_urb(ir->urb_in);
523 static const struct usb_device_id irtoy_table[] = {
524 { USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xfd08, USB_CLASS_CDC_DATA) },
525 { USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xf58b, USB_CLASS_CDC_DATA) },
529 static struct usb_driver irtoy_driver = {
530 .name = KBUILD_MODNAME,
531 .probe = irtoy_probe,
532 .disconnect = irtoy_disconnect,
533 .id_table = irtoy_table,
536 module_usb_driver(irtoy_driver);
538 MODULE_AUTHOR("Sean Young <sean@mess.org>");
539 MODULE_DESCRIPTION("Infrared Toy and IR Droid driver");
540 MODULE_LICENSE("GPL");
541 MODULE_DEVICE_TABLE(usb, irtoy_table);