xref: /openbmc/linux/drivers/media/rc/ir_toy.c (revision 7219a692)
1261463dbSSean Young // SPDX-License-Identifier: GPL-2.0-or-later
2261463dbSSean Young 
3261463dbSSean Young /*
4261463dbSSean Young  * Infrared Toy and IR Droid RC core driver
5261463dbSSean Young  *
6261463dbSSean Young  * Copyright (C) 2020 Sean Young <sean@mess.org>
78bff1386SSean Young  *
88bff1386SSean Young  * http://dangerousprototypes.com/docs/USB_IR_Toy:_Sampling_mode
98bff1386SSean Young  *
10261463dbSSean Young  * This driver is based on the lirc driver which can be found here:
11261463dbSSean Young  * https://sourceforge.net/p/lirc/git/ci/master/tree/plugins/irtoy.c
12261463dbSSean Young  * Copyright (C) 2011 Peter Kooiman <pkooiman@gmail.com>
13261463dbSSean Young  */
14261463dbSSean Young 
15261463dbSSean Young #include <asm/unaligned.h>
16261463dbSSean Young #include <linux/completion.h>
17261463dbSSean Young #include <linux/kernel.h>
18261463dbSSean Young #include <linux/module.h>
19261463dbSSean Young #include <linux/usb.h>
20261463dbSSean Young #include <linux/slab.h>
21261463dbSSean Young #include <linux/usb/input.h>
22261463dbSSean Young 
23261463dbSSean Young #include <media/rc-core.h>
24261463dbSSean Young 
25261463dbSSean Young static const u8 COMMAND_VERSION[] = { 'v' };
26261463dbSSean Young // End transmit and repeat reset command so we exit sump mode
27261463dbSSean Young static const u8 COMMAND_RESET[] = { 0xff, 0xff, 0, 0, 0, 0, 0 };
28261463dbSSean Young static const u8 COMMAND_SMODE_ENTER[] = { 's' };
294114978dSSean Young static const u8 COMMAND_SMODE_EXIT[] = { 0 };
30261463dbSSean Young static const u8 COMMAND_TXSTART[] = { 0x26, 0x24, 0x25, 0x03 };
31261463dbSSean Young 
32261463dbSSean Young #define REPLY_XMITCOUNT 't'
33261463dbSSean Young #define REPLY_XMITSUCCESS 'C'
34261463dbSSean Young #define REPLY_VERSION 'V'
35261463dbSSean Young #define REPLY_SAMPLEMODEPROTO 'S'
36261463dbSSean Young 
37261463dbSSean Young #define TIMEOUT 500
38261463dbSSean Young 
39261463dbSSean Young #define LEN_XMITRES 3
40261463dbSSean Young #define LEN_VERSION 4
41261463dbSSean Young #define LEN_SAMPLEMODEPROTO 3
42261463dbSSean Young 
43261463dbSSean Young #define MIN_FW_VERSION 20
44528222d8SSean Young #define UNIT_US 21
45528222d8SSean Young #define MAX_TIMEOUT_US (UNIT_US * U16_MAX)
46261463dbSSean Young 
47261463dbSSean Young #define MAX_PACKET 64
48261463dbSSean Young 
49261463dbSSean Young enum state {
50261463dbSSean Young 	STATE_IRDATA,
518bff1386SSean Young 	STATE_COMMAND_NO_RESP,
52261463dbSSean Young 	STATE_COMMAND,
53261463dbSSean Young 	STATE_TX,
54261463dbSSean Young };
55261463dbSSean Young 
56261463dbSSean Young struct irtoy {
57261463dbSSean Young 	struct device *dev;
58261463dbSSean Young 	struct usb_device *usbdev;
59261463dbSSean Young 
60261463dbSSean Young 	struct rc_dev *rc;
61261463dbSSean Young 	struct urb *urb_in, *urb_out;
62261463dbSSean Young 
63261463dbSSean Young 	u8 *in;
64261463dbSSean Young 	u8 *out;
65261463dbSSean Young 	struct completion command_done;
66261463dbSSean Young 
67261463dbSSean Young 	bool pulse;
68261463dbSSean Young 	enum state state;
69261463dbSSean Young 
70261463dbSSean Young 	void *tx_buf;
71261463dbSSean Young 	uint tx_len;
72261463dbSSean Young 
73261463dbSSean Young 	uint emitted;
74261463dbSSean Young 	uint hw_version;
75261463dbSSean Young 	uint sw_version;
76261463dbSSean Young 	uint proto_version;
77261463dbSSean Young 
78261463dbSSean Young 	char phys[64];
79261463dbSSean Young };
80261463dbSSean Young 
irtoy_response(struct irtoy * irtoy,u32 len)81261463dbSSean Young static void irtoy_response(struct irtoy *irtoy, u32 len)
82261463dbSSean Young {
83261463dbSSean Young 	switch (irtoy->state) {
84261463dbSSean Young 	case STATE_COMMAND:
85261463dbSSean Young 		if (len == LEN_VERSION && irtoy->in[0] == REPLY_VERSION) {
86261463dbSSean Young 			uint version;
87261463dbSSean Young 
88261463dbSSean Young 			irtoy->in[LEN_VERSION] = 0;
89261463dbSSean Young 
90261463dbSSean Young 			if (kstrtouint(irtoy->in + 1, 10, &version)) {
91261463dbSSean Young 				dev_err(irtoy->dev, "invalid version %*phN. Please make sure you are using firmware v20 or higher",
92261463dbSSean Young 					LEN_VERSION, irtoy->in);
93261463dbSSean Young 				break;
94261463dbSSean Young 			}
95261463dbSSean Young 
96261463dbSSean Young 			dev_dbg(irtoy->dev, "version %s\n", irtoy->in);
97261463dbSSean Young 
98261463dbSSean Young 			irtoy->hw_version = version / 100;
99261463dbSSean Young 			irtoy->sw_version = version % 100;
100261463dbSSean Young 
101261463dbSSean Young 			irtoy->state = STATE_IRDATA;
102261463dbSSean Young 			complete(&irtoy->command_done);
103261463dbSSean Young 		} else if (len == LEN_SAMPLEMODEPROTO &&
104261463dbSSean Young 			   irtoy->in[0] == REPLY_SAMPLEMODEPROTO) {
105261463dbSSean Young 			uint version;
106261463dbSSean Young 
107261463dbSSean Young 			irtoy->in[LEN_SAMPLEMODEPROTO] = 0;
108261463dbSSean Young 
109261463dbSSean Young 			if (kstrtouint(irtoy->in + 1, 10, &version)) {
110261463dbSSean Young 				dev_err(irtoy->dev, "invalid sample mode response %*phN",
111261463dbSSean Young 					LEN_SAMPLEMODEPROTO, irtoy->in);
112261463dbSSean Young 				return;
113261463dbSSean Young 			}
114261463dbSSean Young 
115261463dbSSean Young 			dev_dbg(irtoy->dev, "protocol %s\n", irtoy->in);
116261463dbSSean Young 
117261463dbSSean Young 			irtoy->proto_version = version;
118261463dbSSean Young 
119261463dbSSean Young 			irtoy->state = STATE_IRDATA;
120261463dbSSean Young 			complete(&irtoy->command_done);
121261463dbSSean Young 		} else {
122261463dbSSean Young 			dev_err(irtoy->dev, "unexpected response to command: %*phN\n",
123261463dbSSean Young 				len, irtoy->in);
124261463dbSSean Young 		}
125261463dbSSean Young 		break;
1261d37c854SSean Young 	case STATE_COMMAND_NO_RESP:
127261463dbSSean Young 	case STATE_IRDATA: {
128261463dbSSean Young 		struct ir_raw_event rawir = { .pulse = irtoy->pulse };
129261463dbSSean Young 		__be16 *in = (__be16 *)irtoy->in;
130261463dbSSean Young 		int i;
131261463dbSSean Young 
132261463dbSSean Young 		for (i = 0; i < len / sizeof(__be16); i++) {
133261463dbSSean Young 			u16 v = be16_to_cpu(in[i]);
134261463dbSSean Young 
135261463dbSSean Young 			if (v == 0xffff) {
136261463dbSSean Young 				rawir.pulse = false;
137261463dbSSean Young 			} else {
138528222d8SSean Young 				rawir.duration = v * UNIT_US;
139261463dbSSean Young 				ir_raw_event_store_with_timeout(irtoy->rc,
140261463dbSSean Young 								&rawir);
141261463dbSSean Young 			}
142261463dbSSean Young 
143261463dbSSean Young 			rawir.pulse = !rawir.pulse;
144261463dbSSean Young 		}
145261463dbSSean Young 
146261463dbSSean Young 		irtoy->pulse = rawir.pulse;
147261463dbSSean Young 
148261463dbSSean Young 		ir_raw_event_handle(irtoy->rc);
149261463dbSSean Young 		break;
150261463dbSSean Young 	}
151261463dbSSean Young 	case STATE_TX:
152261463dbSSean Young 		if (irtoy->tx_len == 0) {
153261463dbSSean Young 			if (len == LEN_XMITRES &&
154261463dbSSean Young 			    irtoy->in[0] == REPLY_XMITCOUNT) {
155261463dbSSean Young 				u16 emitted = get_unaligned_be16(irtoy->in + 1);
156261463dbSSean Young 
157261463dbSSean Young 				dev_dbg(irtoy->dev, "emitted:%u\n", emitted);
158261463dbSSean Young 
159261463dbSSean Young 				irtoy->emitted = emitted;
160261463dbSSean Young 			} else if (len == 1 &&
161261463dbSSean Young 				   irtoy->in[0] == REPLY_XMITSUCCESS) {
162261463dbSSean Young 				irtoy->state = STATE_IRDATA;
163261463dbSSean Young 				complete(&irtoy->command_done);
164261463dbSSean Young 			}
165261463dbSSean Young 		} else {
166261463dbSSean Young 			// send next part of tx buffer
167261463dbSSean Young 			uint space = irtoy->in[0];
168261463dbSSean Young 			uint buf_len;
169261463dbSSean Young 			int err;
170261463dbSSean Young 
171261463dbSSean Young 			if (len != 1 || space > MAX_PACKET || space == 0) {
1721d37c854SSean Young 				dev_dbg(irtoy->dev, "packet length expected: %*phN\n",
173261463dbSSean Young 					len, irtoy->in);
174261463dbSSean Young 				break;
175261463dbSSean Young 			}
176261463dbSSean Young 
177261463dbSSean Young 			buf_len = min(space, irtoy->tx_len);
178261463dbSSean Young 
179261463dbSSean Young 			dev_dbg(irtoy->dev, "remaining:%u sending:%u\n",
180261463dbSSean Young 				irtoy->tx_len, buf_len);
181261463dbSSean Young 
182261463dbSSean Young 			memcpy(irtoy->out, irtoy->tx_buf, buf_len);
183261463dbSSean Young 			irtoy->urb_out->transfer_buffer_length = buf_len;
184261463dbSSean Young 			err = usb_submit_urb(irtoy->urb_out, GFP_ATOMIC);
185261463dbSSean Young 			if (err != 0) {
186261463dbSSean Young 				dev_err(irtoy->dev, "fail to submit tx buf urb: %d\n",
187261463dbSSean Young 					err);
188261463dbSSean Young 				irtoy->state = STATE_IRDATA;
189261463dbSSean Young 				complete(&irtoy->command_done);
190261463dbSSean Young 				break;
191261463dbSSean Young 			}
192261463dbSSean Young 
193261463dbSSean Young 			irtoy->tx_buf += buf_len;
194261463dbSSean Young 			irtoy->tx_len -= buf_len;
195261463dbSSean Young 		}
196261463dbSSean Young 		break;
197261463dbSSean Young 	}
198261463dbSSean Young }
199261463dbSSean Young 
irtoy_out_callback(struct urb * urb)200261463dbSSean Young static void irtoy_out_callback(struct urb *urb)
201261463dbSSean Young {
202261463dbSSean Young 	struct irtoy *irtoy = urb->context;
203261463dbSSean Young 
204261463dbSSean Young 	if (urb->status == 0) {
2058bff1386SSean Young 		if (irtoy->state == STATE_COMMAND_NO_RESP)
206261463dbSSean Young 			complete(&irtoy->command_done);
207261463dbSSean Young 	} else {
208261463dbSSean Young 		dev_warn(irtoy->dev, "out urb status: %d\n", urb->status);
209261463dbSSean Young 	}
210261463dbSSean Young }
211261463dbSSean Young 
irtoy_in_callback(struct urb * urb)212261463dbSSean Young static void irtoy_in_callback(struct urb *urb)
213261463dbSSean Young {
214261463dbSSean Young 	struct irtoy *irtoy = urb->context;
215261463dbSSean Young 	int ret;
216261463dbSSean Young 
21792f46151SSean Young 	switch (urb->status) {
21892f46151SSean Young 	case 0:
219261463dbSSean Young 		irtoy_response(irtoy, urb->actual_length);
22092f46151SSean Young 		break;
22192f46151SSean Young 	case -ECONNRESET:
22292f46151SSean Young 	case -ENOENT:
22392f46151SSean Young 	case -ESHUTDOWN:
22492f46151SSean Young 	case -EPROTO:
22592f46151SSean Young 	case -EPIPE:
22692f46151SSean Young 		usb_unlink_urb(urb);
22792f46151SSean Young 		return;
22892f46151SSean Young 	default:
229261463dbSSean Young 		dev_dbg(irtoy->dev, "in urb status: %d\n", urb->status);
23092f46151SSean Young 	}
231261463dbSSean Young 
232261463dbSSean Young 	ret = usb_submit_urb(urb, GFP_ATOMIC);
233261463dbSSean Young 	if (ret && ret != -ENODEV)
234261463dbSSean Young 		dev_warn(irtoy->dev, "failed to resubmit urb: %d\n", ret);
235261463dbSSean Young }
236261463dbSSean Young 
irtoy_command(struct irtoy * irtoy,const u8 * cmd,int cmd_len,enum state state)237261463dbSSean Young static int irtoy_command(struct irtoy *irtoy, const u8 *cmd, int cmd_len,
238261463dbSSean Young 			 enum state state)
239261463dbSSean Young {
240261463dbSSean Young 	int err;
241261463dbSSean Young 
242261463dbSSean Young 	init_completion(&irtoy->command_done);
243261463dbSSean Young 
244261463dbSSean Young 	irtoy->state = state;
245261463dbSSean Young 
246261463dbSSean Young 	memcpy(irtoy->out, cmd, cmd_len);
247261463dbSSean Young 	irtoy->urb_out->transfer_buffer_length = cmd_len;
248261463dbSSean Young 
249261463dbSSean Young 	err = usb_submit_urb(irtoy->urb_out, GFP_KERNEL);
250261463dbSSean Young 	if (err != 0)
251261463dbSSean Young 		return err;
252261463dbSSean Young 
253261463dbSSean Young 	if (!wait_for_completion_timeout(&irtoy->command_done,
254261463dbSSean Young 					 msecs_to_jiffies(TIMEOUT))) {
255261463dbSSean Young 		usb_kill_urb(irtoy->urb_out);
256261463dbSSean Young 		return -ETIMEDOUT;
257261463dbSSean Young 	}
258261463dbSSean Young 
259261463dbSSean Young 	return 0;
260261463dbSSean Young }
261261463dbSSean Young 
irtoy_setup(struct irtoy * irtoy)262261463dbSSean Young static int irtoy_setup(struct irtoy *irtoy)
263261463dbSSean Young {
264261463dbSSean Young 	int err;
265261463dbSSean Young 
266261463dbSSean Young 	err = irtoy_command(irtoy, COMMAND_RESET, sizeof(COMMAND_RESET),
2678bff1386SSean Young 			    STATE_COMMAND_NO_RESP);
268261463dbSSean Young 	if (err != 0) {
269261463dbSSean Young 		dev_err(irtoy->dev, "could not write reset command: %d\n",
270261463dbSSean Young 			err);
271261463dbSSean Young 		return err;
272261463dbSSean Young 	}
273261463dbSSean Young 
274261463dbSSean Young 	usleep_range(50, 50);
275261463dbSSean Young 
276261463dbSSean Young 	// get version
277261463dbSSean Young 	err = irtoy_command(irtoy, COMMAND_VERSION, sizeof(COMMAND_VERSION),
278261463dbSSean Young 			    STATE_COMMAND);
279261463dbSSean Young 	if (err) {
280261463dbSSean Young 		dev_err(irtoy->dev, "could not write version command: %d\n",
281261463dbSSean Young 			err);
282261463dbSSean Young 		return err;
283261463dbSSean Young 	}
284261463dbSSean Young 
285261463dbSSean Young 	// enter sample mode
286261463dbSSean Young 	err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
287261463dbSSean Young 			    sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
288261463dbSSean Young 	if (err)
289261463dbSSean Young 		dev_err(irtoy->dev, "could not write sample command: %d\n",
290261463dbSSean Young 			err);
291261463dbSSean Young 
292261463dbSSean Young 	return err;
293261463dbSSean Young }
294261463dbSSean Young 
295261463dbSSean Young /*
296261463dbSSean Young  * When sending IR, it is imperative that we send the IR data as quickly
297261463dbSSean Young  * as possible to the device, so it does not run out of IR data and
298261463dbSSean Young  * introduce gaps. Allocate the buffer here, and then feed the data from
299261463dbSSean Young  * the urb callback handler.
300261463dbSSean Young  */
irtoy_tx(struct rc_dev * rc,uint * txbuf,uint count)301261463dbSSean Young static int irtoy_tx(struct rc_dev *rc, uint *txbuf, uint count)
302261463dbSSean Young {
303261463dbSSean Young 	struct irtoy *irtoy = rc->priv;
304261463dbSSean Young 	unsigned int i, size;
305261463dbSSean Young 	__be16 *buf;
306261463dbSSean Young 	int err;
307261463dbSSean Young 
308261463dbSSean Young 	size = sizeof(u16) * (count + 1);
309261463dbSSean Young 	buf = kmalloc(size, GFP_KERNEL);
310261463dbSSean Young 	if (!buf)
311261463dbSSean Young 		return -ENOMEM;
312261463dbSSean Young 
313261463dbSSean Young 	for (i = 0; i < count; i++) {
314528222d8SSean Young 		u16 v = DIV_ROUND_CLOSEST(txbuf[i], UNIT_US);
315261463dbSSean Young 
316261463dbSSean Young 		if (!v)
317261463dbSSean Young 			v = 1;
318261463dbSSean Young 		buf[i] = cpu_to_be16(v);
319261463dbSSean Young 	}
320261463dbSSean Young 
321febfe985SSean Young 	buf[count] = cpu_to_be16(0xffff);
322261463dbSSean Young 
323261463dbSSean Young 	irtoy->tx_buf = buf;
324261463dbSSean Young 	irtoy->tx_len = size;
325261463dbSSean Young 	irtoy->emitted = 0;
326261463dbSSean Young 
3274114978dSSean Young 	// There is an issue where if the unit is receiving IR while the
3284114978dSSean Young 	// first TXSTART command is sent, the device might end up hanging
3294114978dSSean Young 	// with its led on. It does not respond to any command when this
3304114978dSSean Young 	// happens. To work around this, re-enter sample mode.
3314114978dSSean Young 	err = irtoy_command(irtoy, COMMAND_SMODE_EXIT,
3324114978dSSean Young 			    sizeof(COMMAND_SMODE_EXIT), STATE_COMMAND_NO_RESP);
3334114978dSSean Young 	if (err) {
3344114978dSSean Young 		dev_err(irtoy->dev, "exit sample mode: %d\n", err);
335*7219a692SZhipeng Lu 		kfree(buf);
3364114978dSSean Young 		return err;
3374114978dSSean Young 	}
3384114978dSSean Young 
3394114978dSSean Young 	err = irtoy_command(irtoy, COMMAND_SMODE_ENTER,
3404114978dSSean Young 			    sizeof(COMMAND_SMODE_ENTER), STATE_COMMAND);
3414114978dSSean Young 	if (err) {
3424114978dSSean Young 		dev_err(irtoy->dev, "enter sample mode: %d\n", err);
343*7219a692SZhipeng Lu 		kfree(buf);
3444114978dSSean Young 		return err;
3454114978dSSean Young 	}
3464114978dSSean Young 
347261463dbSSean Young 	err = irtoy_command(irtoy, COMMAND_TXSTART, sizeof(COMMAND_TXSTART),
348261463dbSSean Young 			    STATE_TX);
349261463dbSSean Young 	kfree(buf);
350261463dbSSean Young 
351261463dbSSean Young 	if (err) {
352261463dbSSean Young 		dev_err(irtoy->dev, "failed to send tx start command: %d\n",
353261463dbSSean Young 			err);
354261463dbSSean Young 		// not sure what state the device is in, reset it
355261463dbSSean Young 		irtoy_setup(irtoy);
356261463dbSSean Young 		return err;
357261463dbSSean Young 	}
358261463dbSSean Young 
359261463dbSSean Young 	if (size != irtoy->emitted) {
360261463dbSSean Young 		dev_err(irtoy->dev, "expected %u emitted, got %u\n", size,
361261463dbSSean Young 			irtoy->emitted);
362261463dbSSean Young 		// not sure what state the device is in, reset it
363261463dbSSean Young 		irtoy_setup(irtoy);
364261463dbSSean Young 		return -EINVAL;
365261463dbSSean Young 	}
366261463dbSSean Young 
367261463dbSSean Young 	return count;
368261463dbSSean Young }
369261463dbSSean Young 
irtoy_tx_carrier(struct rc_dev * rc,uint32_t carrier)3708bff1386SSean Young static int irtoy_tx_carrier(struct rc_dev *rc, uint32_t carrier)
3718bff1386SSean Young {
3728bff1386SSean Young 	struct irtoy *irtoy = rc->priv;
3738bff1386SSean Young 	u8 buf[3];
3748bff1386SSean Young 	int err;
3758bff1386SSean Young 
3768bff1386SSean Young 	if (carrier < 11800)
3778bff1386SSean Young 		return -EINVAL;
3788bff1386SSean Young 
3798bff1386SSean Young 	buf[0] = 0x06;
3808bff1386SSean Young 	buf[1] = DIV_ROUND_CLOSEST(48000000, 16 * carrier) - 1;
3818bff1386SSean Young 	buf[2] = 0;
3828bff1386SSean Young 
3838bff1386SSean Young 	err = irtoy_command(irtoy, buf, sizeof(buf), STATE_COMMAND_NO_RESP);
3848bff1386SSean Young 	if (err)
3858bff1386SSean Young 		dev_err(irtoy->dev, "could not write carrier command: %d\n",
3868bff1386SSean Young 			err);
3878bff1386SSean Young 
3888bff1386SSean Young 	return err;
3898bff1386SSean Young }
3908bff1386SSean Young 
irtoy_probe(struct usb_interface * intf,const struct usb_device_id * id)391261463dbSSean Young static int irtoy_probe(struct usb_interface *intf,
392261463dbSSean Young 		       const struct usb_device_id *id)
393261463dbSSean Young {
394261463dbSSean Young 	struct usb_host_interface *idesc = intf->cur_altsetting;
395261463dbSSean Young 	struct usb_device *usbdev = interface_to_usbdev(intf);
396261463dbSSean Young 	struct usb_endpoint_descriptor *ep_in = NULL;
397261463dbSSean Young 	struct usb_endpoint_descriptor *ep_out = NULL;
398261463dbSSean Young 	struct usb_endpoint_descriptor *ep = NULL;
399261463dbSSean Young 	struct irtoy *irtoy;
400261463dbSSean Young 	struct rc_dev *rc;
401261463dbSSean Young 	struct urb *urb;
402261463dbSSean Young 	int i, pipe, err = -ENOMEM;
403261463dbSSean Young 
404261463dbSSean Young 	for (i = 0; i < idesc->desc.bNumEndpoints; i++) {
405261463dbSSean Young 		ep = &idesc->endpoint[i].desc;
406261463dbSSean Young 
407261463dbSSean Young 		if (!ep_in && usb_endpoint_is_bulk_in(ep) &&
408261463dbSSean Young 		    usb_endpoint_maxp(ep) == MAX_PACKET)
409261463dbSSean Young 			ep_in = ep;
410261463dbSSean Young 
411261463dbSSean Young 		if (!ep_out && usb_endpoint_is_bulk_out(ep) &&
412261463dbSSean Young 		    usb_endpoint_maxp(ep) == MAX_PACKET)
413261463dbSSean Young 			ep_out = ep;
414261463dbSSean Young 	}
415261463dbSSean Young 
416261463dbSSean Young 	if (!ep_in || !ep_out) {
417261463dbSSean Young 		dev_err(&intf->dev, "required endpoints not found\n");
418261463dbSSean Young 		return -ENODEV;
419261463dbSSean Young 	}
420261463dbSSean Young 
421261463dbSSean Young 	irtoy = kzalloc(sizeof(*irtoy), GFP_KERNEL);
422261463dbSSean Young 	if (!irtoy)
423261463dbSSean Young 		return -ENOMEM;
424261463dbSSean Young 
425261463dbSSean Young 	irtoy->in = kmalloc(MAX_PACKET,  GFP_KERNEL);
426261463dbSSean Young 	if (!irtoy->in)
427261463dbSSean Young 		goto free_irtoy;
428261463dbSSean Young 
429261463dbSSean Young 	irtoy->out = kmalloc(MAX_PACKET,  GFP_KERNEL);
430261463dbSSean Young 	if (!irtoy->out)
431261463dbSSean Young 		goto free_irtoy;
432261463dbSSean Young 
433261463dbSSean Young 	rc = rc_allocate_device(RC_DRIVER_IR_RAW);
434261463dbSSean Young 	if (!rc)
435261463dbSSean Young 		goto free_irtoy;
436261463dbSSean Young 
437261463dbSSean Young 	urb = usb_alloc_urb(0, GFP_KERNEL);
438261463dbSSean Young 	if (!urb)
439261463dbSSean Young 		goto free_rcdev;
440261463dbSSean Young 
441261463dbSSean Young 	pipe = usb_rcvbulkpipe(usbdev, ep_in->bEndpointAddress);
442261463dbSSean Young 	usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->in, MAX_PACKET,
443261463dbSSean Young 			  irtoy_in_callback, irtoy);
444261463dbSSean Young 	irtoy->urb_in = urb;
445261463dbSSean Young 
446261463dbSSean Young 	urb = usb_alloc_urb(0, GFP_KERNEL);
447261463dbSSean Young 	if (!urb)
448261463dbSSean Young 		goto free_rcdev;
449261463dbSSean Young 
450261463dbSSean Young 	pipe = usb_sndbulkpipe(usbdev, ep_out->bEndpointAddress);
451261463dbSSean Young 	usb_fill_bulk_urb(urb, usbdev, pipe, irtoy->out, MAX_PACKET,
452261463dbSSean Young 			  irtoy_out_callback, irtoy);
453261463dbSSean Young 
454261463dbSSean Young 	irtoy->dev = &intf->dev;
455261463dbSSean Young 	irtoy->usbdev = usbdev;
456261463dbSSean Young 	irtoy->rc = rc;
457261463dbSSean Young 	irtoy->urb_out = urb;
458261463dbSSean Young 	irtoy->pulse = true;
459261463dbSSean Young 
460261463dbSSean Young 	err = usb_submit_urb(irtoy->urb_in, GFP_KERNEL);
461261463dbSSean Young 	if (err != 0) {
462261463dbSSean Young 		dev_err(irtoy->dev, "fail to submit in urb: %d\n", err);
46352cdb013SPeiwei Hu 		goto free_rcdev;
464261463dbSSean Young 	}
465261463dbSSean Young 
466261463dbSSean Young 	err = irtoy_setup(irtoy);
467261463dbSSean Young 	if (err)
468261463dbSSean Young 		goto free_rcdev;
469261463dbSSean Young 
4705173cca0SSean Young 	dev_info(irtoy->dev, "version: hardware %u, firmware %u.%u, protocol %u",
4715173cca0SSean Young 		 irtoy->hw_version, irtoy->sw_version / 10,
4725173cca0SSean Young 		 irtoy->sw_version % 10, irtoy->proto_version);
473261463dbSSean Young 
474261463dbSSean Young 	if (irtoy->sw_version < MIN_FW_VERSION) {
475261463dbSSean Young 		dev_err(irtoy->dev, "need firmware V%02u or higher",
476261463dbSSean Young 			MIN_FW_VERSION);
477261463dbSSean Young 		err = -ENODEV;
478261463dbSSean Young 		goto free_rcdev;
479261463dbSSean Young 	}
480261463dbSSean Young 
481261463dbSSean Young 	usb_make_path(usbdev, irtoy->phys, sizeof(irtoy->phys));
482261463dbSSean Young 
483261463dbSSean Young 	rc->device_name = "Infrared Toy";
484261463dbSSean Young 	rc->driver_name = KBUILD_MODNAME;
485261463dbSSean Young 	rc->input_phys = irtoy->phys;
486261463dbSSean Young 	usb_to_input_id(usbdev, &rc->input_id);
487261463dbSSean Young 	rc->dev.parent = &intf->dev;
488261463dbSSean Young 	rc->priv = irtoy;
489261463dbSSean Young 	rc->tx_ir = irtoy_tx;
4908bff1386SSean Young 	rc->s_tx_carrier = irtoy_tx_carrier;
491261463dbSSean Young 	rc->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER;
492261463dbSSean Young 	rc->map_name = RC_MAP_RC6_MCE;
493528222d8SSean Young 	rc->rx_resolution = UNIT_US;
494261463dbSSean Young 	rc->timeout = IR_DEFAULT_TIMEOUT;
495261463dbSSean Young 
496261463dbSSean Young 	/*
497261463dbSSean Young 	 * end of transmission is detected by absence of a usb packet
498261463dbSSean Young 	 * with more pulse/spaces. However, each usb packet sent can
499261463dbSSean Young 	 * contain 32 pulse/spaces, which can be quite lengthy, so there
500261463dbSSean Young 	 * can be a delay between usb packets. For example with nec there is a
501261463dbSSean Young 	 * 17ms gap between packets.
502261463dbSSean Young 	 *
503261463dbSSean Young 	 * So, make timeout a largish minimum which works with most protocols.
504261463dbSSean Young 	 */
505528222d8SSean Young 	rc->min_timeout = MS_TO_US(40);
506528222d8SSean Young 	rc->max_timeout = MAX_TIMEOUT_US;
507261463dbSSean Young 
508261463dbSSean Young 	err = rc_register_device(rc);
509261463dbSSean Young 	if (err)
510261463dbSSean Young 		goto free_rcdev;
511261463dbSSean Young 
512261463dbSSean Young 	usb_set_intfdata(intf, irtoy);
513261463dbSSean Young 
514261463dbSSean Young 	return 0;
515261463dbSSean Young 
516261463dbSSean Young free_rcdev:
517261463dbSSean Young 	usb_kill_urb(irtoy->urb_out);
518261463dbSSean Young 	usb_free_urb(irtoy->urb_out);
519261463dbSSean Young 	usb_kill_urb(irtoy->urb_in);
520261463dbSSean Young 	usb_free_urb(irtoy->urb_in);
521261463dbSSean Young 	rc_free_device(rc);
522261463dbSSean Young free_irtoy:
523261463dbSSean Young 	kfree(irtoy->in);
524261463dbSSean Young 	kfree(irtoy->out);
525261463dbSSean Young 	kfree(irtoy);
526261463dbSSean Young 	return err;
527261463dbSSean Young }
528261463dbSSean Young 
irtoy_disconnect(struct usb_interface * intf)529261463dbSSean Young static void irtoy_disconnect(struct usb_interface *intf)
530261463dbSSean Young {
531261463dbSSean Young 	struct irtoy *ir = usb_get_intfdata(intf);
532261463dbSSean Young 
533261463dbSSean Young 	rc_unregister_device(ir->rc);
534261463dbSSean Young 	usb_set_intfdata(intf, NULL);
535261463dbSSean Young 	usb_kill_urb(ir->urb_out);
536261463dbSSean Young 	usb_free_urb(ir->urb_out);
537261463dbSSean Young 	usb_kill_urb(ir->urb_in);
538261463dbSSean Young 	usb_free_urb(ir->urb_in);
539261463dbSSean Young 	kfree(ir->in);
540261463dbSSean Young 	kfree(ir->out);
541261463dbSSean Young 	kfree(ir);
542261463dbSSean Young }
543261463dbSSean Young 
544261463dbSSean Young static const struct usb_device_id irtoy_table[] = {
545261463dbSSean Young 	{ USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xfd08, USB_CLASS_CDC_DATA) },
5464487e021SSean Young 	{ USB_DEVICE_INTERFACE_CLASS(0x04d8, 0xf58b, USB_CLASS_CDC_DATA) },
547261463dbSSean Young 	{ }
548261463dbSSean Young };
549261463dbSSean Young 
550261463dbSSean Young static struct usb_driver irtoy_driver = {
551261463dbSSean Young 	.name = KBUILD_MODNAME,
552261463dbSSean Young 	.probe = irtoy_probe,
553261463dbSSean Young 	.disconnect = irtoy_disconnect,
554261463dbSSean Young 	.id_table = irtoy_table,
555261463dbSSean Young };
556261463dbSSean Young 
557261463dbSSean Young module_usb_driver(irtoy_driver);
558261463dbSSean Young 
559261463dbSSean Young MODULE_AUTHOR("Sean Young <sean@mess.org>");
560261463dbSSean Young MODULE_DESCRIPTION("Infrared Toy and IR Droid driver");
561261463dbSSean Young MODULE_LICENSE("GPL");
562261463dbSSean Young MODULE_DEVICE_TABLE(usb, irtoy_table);
563