1
2
3
4
5
6
7
8
9
10
11
12
13
14
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>
22
23#include <media/rc-core.h>
24
25static const u8 COMMAND_VERSION[] = { 'v' };
26
27static const u8 COMMAND_RESET[] = { 0xff, 0xff, 0, 0, 0, 0, 0 };
28static const u8 COMMAND_SMODE_ENTER[] = { 's' };
29static const u8 COMMAND_SMODE_EXIT[] = { 0 };
30static const u8 COMMAND_TXSTART[] = { 0x26, 0x24, 0x25, 0x03 };
31
32#define REPLY_XMITCOUNT 't'
33#define REPLY_XMITSUCCESS 'C'
34#define REPLY_VERSION 'V'
35#define REPLY_SAMPLEMODEPROTO 'S'
36
37#define TIMEOUT 500
38
39#define LEN_XMITRES 3
40#define LEN_VERSION 4
41#define LEN_SAMPLEMODEPROTO 3
42
43#define MIN_FW_VERSION 20
44#define UNIT_US 21
45#define MAX_TIMEOUT_US (UNIT_US * U16_MAX)
46
47#define MAX_PACKET 64
48
49enum state {
50 STATE_IRDATA,
51 STATE_COMMAND_NO_RESP,
52 STATE_COMMAND,
53 STATE_TX,
54};
55
56struct irtoy {
57 struct device *dev;
58 struct usb_device *usbdev;
59
60 struct rc_dev *rc;
61 struct urb *urb_in, *urb_out;
62
63 u8 *in;
64 u8 *out;
65 struct completion command_done;
66
67 bool pulse;
68 enum state state;
69
70 void *tx_buf;
71 uint tx_len;
72
73 uint emitted;
74 uint hw_version;
75 uint sw_version;
76 uint proto_version;
77
78 char phys[64];
79};
80
81static void irtoy_response(struct irtoy *irtoy, u32 len)
82{
83 switch (irtoy->state) {
84 case STATE_COMMAND:
85 if (len == LEN_VERSION && irtoy->in[0] == REPLY_VERSION) {
86 uint version;
87
88 irtoy->in[LEN_VERSION] = 0;
89
90 if (kstrtouint(irtoy->in + 1, 10, &version)) {
91 dev_err(irtoy->dev, "invalid version %*phN. Please make sure you are using firmware v20 or higher",
92 LEN_VERSION, irtoy->in);
93 break;
94 }
95
96 dev_dbg(irtoy->dev, "version %s\n", irtoy->in);
97
98 irtoy->hw_version = version / 100;
99 irtoy->sw_version = version % 100;
100
101 irtoy->state = STATE_IRDATA;
102 complete(&irtoy->command_done);
103 } else if (len == LEN_SAMPLEMODEPROTO &&
104 irtoy->in[0] == REPLY_SAMPLEMODEPROTO) {
105 uint version;
106
107 irtoy->in[LEN_SAMPLEMODEPROTO] = 0;
108
109 if (kstrtouint(irtoy->in + 1, 10, &version)) {
110 dev_err(irtoy->dev, "invalid sample mode response %*phN",
111 LEN_SAMPLEMODEPROTO, irtoy->in);
112 return;
113 }
114
115 dev_dbg(irtoy->dev, "protocol %s\n", irtoy->in);
116
117 irtoy->proto_version = version;
118
119 irtoy->state = STATE_IRDATA;
120 complete(&irtoy->command_done);
121 } else {
122 dev_err(irtoy->dev, "unexpected response to command: %*phN\n",
123 len, irtoy->in);
124 }
125 break;
126 case STATE_COMMAND_NO_RESP:
127 case STATE_IRDATA: {
128 struct ir_raw_event rawir = { .pulse = irtoy->pulse };
129 __be16 *in = (__be16 *)irtoy->in;
130 int i;
131
132 for (i = 0; i < len / sizeof(__be16); i++) {
133 u16 v = be16_to_cpu(in[i]);
134
135 if (v == 0xffff) {
136 rawir.pulse = false;
137 } else {
138 rawir.duration = v * UNIT_US;
139 ir_raw_event_store_with_timeout(irtoy->rc,
140 &rawir);
141 }
142
143 rawir.pulse = !rawir.pulse;
144 }
145
146 irtoy->pulse = rawir.pulse;
147
148 ir_raw_event_handle(irtoy->rc);
149 break;
150 }
151 case STATE_TX:
152 if (irtoy->tx_len == 0) {
153 if (len == LEN_XMITRES &&
154 irtoy->in[0] == REPLY_XMITCOUNT) {
155 u16 emitted = get_unaligned_be16(irtoy->in + 1);
156
157 dev_dbg(irtoy->dev, "emitted:%u\n", emitted);
158
159 irtoy->emitted = emitted;
160 } else if (len == 1 &&
161 irtoy->in[0] == REPLY_XMITSUCCESS) {
162 irtoy->state = STATE_IRDATA;
163 complete(&irtoy->command_done);
164 }
165 } else {
166
167 uint space = irtoy->in[0];
168 uint buf_len;
169 int err;
170
171 if (len != 1 || space > MAX_PACKET || space == 0) {
172 dev_dbg(irtoy->dev, "packet length expected: %*phN\n",
173 len, irtoy->in);
174 break;
175 }
176
177 buf_len = min(space, irtoy->tx_len);
178
179 dev_dbg(irtoy->dev, "remaining:%u sending:%u\n",
180 irtoy->tx_len, buf_len);
181
182 memcpy(irtoy->out, irtoy->tx_buf, buf_len);
183 irtoy->urb_out->transfer_buffer_length = buf_len;
184 err = usb_submit_urb(irtoy->urb_out, GFP_ATOMIC);
185 if (err != 0) {
186 dev_err(irtoy->dev, "fail to submit tx buf urb: %d\n",
187 err);
188 irtoy->state = STATE_IRDATA;
189 complete(&irtoy->command_done);
190 break;
191 }
192
193 irtoy->tx_buf += buf_len;
194 irtoy->tx_len -= buf_len;
195 }
196 break;
197 }
198}
199
200static void irtoy_out_callback(struct urb *urb)
201{
202 struct irtoy *irtoy = urb->context;
203
204 if (urb->status == 0) {
205 if (irtoy->state == STATE_COMMAND_NO_RESP)
206 complete(&irtoy->command_done);
207 } else {
208 dev_warn(irtoy->dev, "out urb status: %d\n", urb->status);
209 }
210}
211
212static void irtoy_in_callback(struct urb *urb)
213{
214 struct irtoy *irtoy = urb->context;
215 int ret;
216
217 switch (urb->status) {
218 case 0:
219 irtoy_response(irtoy, urb->actual_length);
220 break;
221 case -ECONNRESET:
222 case -ENOENT:
223 case -ESHUTDOWN:
224 case -EPROTO:
225 case -EPIPE:
226 usb_unlink_urb(urb);
227 return;
228 default:
229 dev_dbg(irtoy->dev, "in urb status: %d\n", urb->status);
230 }
231
232 ret = usb_submit_urb(urb, GFP_ATOMIC);
233 if (ret && ret != -ENODEV)
234 dev_warn(irtoy->dev, "failed to resubmit urb: %d\n", ret);
235}
236
237static int irtoy_command(struct irtoy *irtoy, const u8 *cmd, int cmd_len,
238 enum state state)
239{
240 int err;
241
242 init_completion(&irtoy->command_done);
243
244 irtoy->state = state;
245
246 memcpy(irtoy->out, cmd, cmd_len);
247 irtoy->urb_out->transfer_buffer_length = cmd_len;
248
249 err = usb_submit_urb(irtoy->urb_out, GFP_KERNEL);
250 if (err != 0)
251 return err;
252
253 if (!wait_for_completion_timeout(&irtoy->command_done,
254 msecs_to_jiffies(TIMEOUT))) {
255 usb_kill_urb(irtoy->urb_out);
256 return -ETIMEDOUT;
257 }
258
259 return 0;
260}
261
262static int irtoy_setup(struct irtoy *irtoy)
263{
264 int err;
265
266 err = irtoy_command(irtoy, COMMAND_RESET, sizeof(COMMAND_RESET),
267 STATE_COMMAND_NO_RESP);
268 if (err != 0) {
269 dev_err(irtoy->dev, "could not write reset command: %d\n",
270 err);
271 return err;
272 }
273
274 usleep_range(50, 50);
275
276
277 err = irtoy_command(irtoy, COMMAND_VERSION, sizeof(COMMAND_VERSION),
278 STATE_COMMAND);
279 if (err) {
280 dev_err(irtoy->dev, "could not write version command: %d\n",
281 err);
282 return err;
283 }
284
285
286 err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
287 sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
288 if (err)
289 dev_err(irtoy->dev, "could not write sample command: %d\n",
290 err);
291
292 return err;
293}
294
295
296
297
298
299
300
301static int irtoy_tx(struct rc_dev *rc, uint *txbuf, uint count)
302{
303 struct irtoy *irtoy = rc->priv;
304 unsigned int i, size;
305 __be16 *buf;
306 int err;
307
308 size = sizeof(u16) * (count + 1);
309 buf = kmalloc(size, GFP_KERNEL);
310 if (!buf)
311 return -ENOMEM;
312
313 for (i = 0; i < count; i++) {
314 u16 v = DIV_ROUND_CLOSEST(txbuf[i], UNIT_US);
315
316 if (!v)
317 v = 1;
318 buf[i] = cpu_to_be16(v);
319 }
320
321 buf[count] = cpu_to_be16(0xffff);
322
323 irtoy->tx_buf = buf;
324 irtoy->tx_len = size;
325 irtoy->emitted = 0;
326
327
328
329
330
331 err = irtoy_command(irtoy, COMMAND_SMODE_EXIT,
332 sizeof(COMMAND_SMODE_EXIT), STATE_COMMAND_NO_RESP);
333 if (err) {
334 dev_err(irtoy->dev, "exit sample mode: %d\n", err);
335 return err;
336 }
337
338 err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
339 sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
340 if (err) {
341 dev_err(irtoy->dev, "enter sample mode: %d\n", err);
342 return err;
343 }
344
345 err = irtoy_command(irtoy, COMMAND_TXSTART, sizeof(COMMAND_TXSTART),
346 STATE_TX);
347 kfree(buf);
348
349 if (err) {
350 dev_err(irtoy->dev, "failed to send tx start command: %d\n",
351 err);
352
353 irtoy_setup(irtoy);
354 return err;
355 }
356
357 if (size != irtoy->emitted) {
358 dev_err(irtoy->dev, "expected %u emitted, got %u\n", size,
359 irtoy->emitted);
360
361 irtoy_setup(irtoy);
362 return -EINVAL;
363 }
364
365 return count;
366}
367
368static int irtoy_tx_carrier(struct rc_dev *rc, uint32_t carrier)
369{
370 struct irtoy *irtoy = rc->priv;
371 u8 buf[3];
372 int err;
373
374 if (carrier < 11800)
375 return -EINVAL;
376
377 buf[0] = 0x06;
378 buf[1] = DIV_ROUND_CLOSEST(48000000, 16 * carrier) - 1;
379 buf[2] = 0;
380
381 err = irtoy_command(irtoy, buf, sizeof(buf), STATE_COMMAND_NO_RESP);
382 if (err)
383 dev_err(irtoy->dev, "could not write carrier command: %d\n",
384 err);
385
386 return err;
387}
388
389static int irtoy_probe(struct usb_interface *intf,
390 const struct usb_device_id *id)
391{
392 struct usb_host_interface *idesc = intf->cur_altsetting;
393 struct usb_device *usbdev = interface_to_usbdev(intf);
394 struct usb_endpoint_descriptor *ep_in = NULL;
395 struct usb_endpoint_descriptor *ep_out = NULL;
396 struct usb_endpoint_descriptor *ep = NULL;
397 struct irtoy *irtoy;
398 struct rc_dev *rc;
399 struct urb *urb;
400 int i, pipe, err = -ENOMEM;
401
402 for (i = 0; i < idesc->desc.bNumEndpoints; i++) {
403 ep = &idesc->endpoint[i].desc;
404
405 if (!ep_in && usb_endpoint_is_bulk_in(ep) &&
406 usb_endpoint_maxp(ep) == MAX_PACKET)
407 ep_in = ep;
408
409 if (!ep_out && usb_endpoint_is_bulk_out(ep) &&
410 usb_endpoint_maxp(ep) == MAX_PACKET)
411 ep_out = ep;
412 }
413
414 if (!ep_in || !ep_out) {
415 dev_err(&intf->dev, "required endpoints not found\n");
416 return -ENODEV;
417 }
418
419 irtoy = kzalloc(sizeof(*irtoy), GFP_KERNEL);
420 if (!irtoy)
421 return -ENOMEM;
422
423 irtoy->in = kmalloc(MAX_PACKET, GFP_KERNEL);
424 if (!irtoy->in)
425 goto free_irtoy;
426
427 irtoy->out = kmalloc(MAX_PACKET, GFP_KERNEL);
428 if (!irtoy->out)
429 goto free_irtoy;
430
431 rc = rc_allocate_device(RC_DRIVER_IR_RAW);
432 if (!rc)
433 goto free_irtoy;
434
435 urb = usb_alloc_urb(0, GFP_KERNEL);
436 if (!urb)
437 goto free_rcdev;
438
439 pipe = usb_rcvbulkpipe(usbdev, ep_in->bEndpointAddress);
440 usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->in, MAX_PACKET,
441 irtoy_in_callback, irtoy);
442 irtoy->urb_in = urb;
443
444 urb = usb_alloc_urb(0, GFP_KERNEL);
445 if (!urb)
446 goto free_rcdev;
447
448 pipe = usb_sndbulkpipe(usbdev, ep_out->bEndpointAddress);
449 usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->out, MAX_PACKET,
450 irtoy_out_callback, irtoy);
451
452 irtoy->dev = &intf->dev;
453 irtoy->usbdev = usbdev;
454 irtoy->rc = rc;
455 irtoy->urb_out = urb;
456 irtoy->pulse = true;
457
458 err = usb_submit_urb(irtoy->urb_in, GFP_KERNEL);
459 if (err != 0) {
460 dev_err(irtoy->dev, "fail to submit in urb: %d\n", err);
461 return err;
462 }
463
464 err = irtoy_setup(irtoy);
465 if (err)
466 goto free_rcdev;
467
468 dev_info(irtoy->dev, "version: hardware %u, firmware %u.%u, protocol %u",
469 irtoy->hw_version, irtoy->sw_version / 10,
470 irtoy->sw_version % 10, irtoy->proto_version);
471
472 if (irtoy->sw_version < MIN_FW_VERSION) {
473 dev_err(irtoy->dev, "need firmware V%02u or higher",
474 MIN_FW_VERSION);
475 err = -ENODEV;
476 goto free_rcdev;
477 }
478
479 usb_make_path(usbdev, irtoy->phys, sizeof(irtoy->phys));
480
481 rc->device_name = "Infrared Toy";
482 rc->driver_name = KBUILD_MODNAME;
483 rc->input_phys = irtoy->phys;
484 usb_to_input_id(usbdev, &rc->input_id);
485 rc->dev.parent = &intf->dev;
486 rc->priv = irtoy;
487 rc->tx_ir = irtoy_tx;
488 rc->s_tx_carrier = irtoy_tx_carrier;
489 rc->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER;
490 rc->map_name = RC_MAP_RC6_MCE;
491 rc->rx_resolution = UNIT_US;
492 rc->timeout = IR_DEFAULT_TIMEOUT;
493
494
495
496
497
498
499
500
501
502
503 rc->min_timeout = MS_TO_US(40);
504 rc->max_timeout = MAX_TIMEOUT_US;
505
506 err = rc_register_device(rc);
507 if (err)
508 goto free_rcdev;
509
510 usb_set_intfdata(intf, irtoy);
511
512 return 0;
513
514free_rcdev:
515 usb_kill_urb(irtoy->urb_out);
516 usb_free_urb(irtoy->urb_out);
517 usb_kill_urb(irtoy->urb_in);
518 usb_free_urb(irtoy->urb_in);
519 rc_free_device(rc);
520free_irtoy:
521 kfree(irtoy->in);
522 kfree(irtoy->out);
523 kfree(irtoy);
524 return err;
525}
526
527static void irtoy_disconnect(struct usb_interface *intf)
528{
529 struct irtoy *ir = usb_get_intfdata(intf);
530
531 rc_unregister_device(ir->rc);
532 usb_set_intfdata(intf, NULL);
533 usb_kill_urb(ir->urb_out);
534 usb_free_urb(ir->urb_out);
535 usb_kill_urb(ir->urb_in);
536 usb_free_urb(ir->urb_in);
537 kfree(ir->in);
538 kfree(ir->out);
539 kfree(ir);
540}
541
542static const struct usb_device_id irtoy_table[] = {
543 { USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xfd08, USB_CLASS_CDC_DATA) },
544 { USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xf58b, USB_CLASS_CDC_DATA) },
545 { }
546};
547
548static struct usb_driver irtoy_driver = {
549 .name = KBUILD_MODNAME,
550 .probe = irtoy_probe,
551 .disconnect = irtoy_disconnect,
552 .id_table = irtoy_table,
553};
554
555module_usb_driver(irtoy_driver);
556
557MODULE_AUTHOR("Sean Young <sean@mess.org>");
558MODULE_DESCRIPTION("Infrared Toy and IR Droid driver");
559MODULE_LICENSE("GPL");
560MODULE_DEVICE_TABLE(usb, irtoy_table);
561