[go: nahoru, domu]

1/*
2 * lirc_igorplugusb - USB remote support for LIRC
3 *
4 * Supports the standard homebrew IgorPlugUSB receiver with Igor's firmware.
5 * See http://www.cesko.host.sk/IgorPlugUSB/IgorPlug-USB%20(AVR)_eng.htm
6 *
7 * The device can only record bursts of up to 36 pulses/spaces.
8 * Works fine with RC5. Longer commands lead to device buffer overrun.
9 * (Maybe a better firmware or a microcontroller with more ram can help?)
10 *
11 * Version 0.1  [beta status]
12 *
13 * Copyright (C) 2004 Jan M. Hochstein
14 *	<hochstein@algo.informatik.tu-darmstadt.de>
15 *
16 * This driver was derived from:
17 *   Paul Miller <pmiller9@users.sourceforge.net>
18 *      "lirc_atiusb" module
19 *   Vladimir Dergachev <volodya@minspring.com>'s 2002
20 *      "USB ATI Remote support" (input device)
21 *   Adrian Dewhurst <sailor-lk@sailorfrag.net>'s 2002
22 *      "USB StreamZap remote driver" (LIRC)
23 *   Artur Lipowski <alipowski@kki.net.pl>'s 2002
24 *      "lirc_dev" and "lirc_gpio" LIRC modules
25 */
26
27/*
28 * This program is free software; you can redistribute it and/or modify
29 * it under the terms of the GNU General Public License as published by
30 * the Free Software Foundation; either version 2 of the License, or
31 * (at your option) any later version.
32 *
33 * This program is distributed in the hope that it will be useful,
34 * but WITHOUT ANY WARRANTY; without even the implied warranty of
35 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
36 * GNU General Public License for more details.
37 *
38 * You should have received a copy of the GNU General Public License
39 * along with this program; if not, write to the Free Software
40 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
41 */
42
43#include <linux/module.h>
44#include <linux/kernel.h>
45#include <linux/kmod.h>
46#include <linux/sched.h>
47#include <linux/errno.h>
48#include <linux/fs.h>
49#include <linux/usb.h>
50#include <linux/time.h>
51
52#include <media/lirc.h>
53#include <media/lirc_dev.h>
54
55
56/* module identification */
57#define DRIVER_VERSION		"0.2"
58#define DRIVER_AUTHOR		\
59	"Jan M. Hochstein <hochstein@algo.informatik.tu-darmstadt.de>"
60#define DRIVER_DESC		"Igorplug USB remote driver for LIRC"
61#define DRIVER_NAME		"lirc_igorplugusb"
62
63/* One mode2 pulse/space has 4 bytes. */
64#define CODE_LENGTH	     sizeof(int)
65
66/* Igor's firmware cannot record bursts longer than 36. */
67#define DEVICE_BUFLEN	   36
68
69/*
70 * Header at the beginning of the device's buffer:
71 *	unsigned char data_length
72 *	unsigned char data_start    (!=0 means ring-buffer overrun)
73 *	unsigned char counter       (incremented by each burst)
74 */
75#define DEVICE_HEADERLEN	3
76
77/* This is for the gap */
78#define ADDITIONAL_LIRC_BYTES   2
79
80/* times to poll per second */
81#define SAMPLE_RATE	     100
82static int sample_rate = SAMPLE_RATE;
83
84
85/**** Igor's USB Request Codes */
86
87#define SET_INFRABUFFER_EMPTY   1
88/**
89 * Params: none
90 * Answer: empty
91 */
92
93#define GET_INFRACODE	   2
94/**
95 * Params:
96 *   wValue: offset to begin reading infra buffer
97 *
98 * Answer: infra data
99 */
100
101#define SET_DATAPORT_DIRECTION  3
102/**
103 * Params:
104 *   wValue: (byte) 1 bit for each data port pin (0=in, 1=out)
105 *
106 * Answer: empty
107 */
108
109#define GET_DATAPORT_DIRECTION  4
110/**
111 * Params: none
112 *
113 * Answer: (byte) 1 bit for each data port pin (0=in, 1=out)
114 */
115
116#define SET_OUT_DATAPORT	5
117/**
118 * Params:
119 *   wValue: byte to write to output data port
120 *
121 * Answer: empty
122 */
123
124#define GET_OUT_DATAPORT	6
125/**
126 * Params: none
127 *
128 * Answer: least significant 3 bits read from output data port
129 */
130
131#define GET_IN_DATAPORT	 7
132/**
133 * Params: none
134 *
135 * Answer: least significant 3 bits read from input data port
136 */
137
138#define READ_EEPROM	     8
139/**
140 * Params:
141 *   wValue: offset to begin reading EEPROM
142 *
143 * Answer: EEPROM bytes
144 */
145
146#define WRITE_EEPROM	    9
147/**
148 * Params:
149 *   wValue: offset to EEPROM byte
150 *   wIndex: byte to write
151 *
152 * Answer: empty
153 */
154
155#define SEND_RS232	      10
156/**
157 * Params:
158 *   wValue: byte to send
159 *
160 * Answer: empty
161 */
162
163#define RECV_RS232	      11
164/**
165 * Params: none
166 *
167 * Answer: byte received
168 */
169
170#define SET_RS232_BAUD	  12
171/**
172 * Params:
173 *   wValue: byte to write to UART bit rate register (UBRR)
174 *
175 * Answer: empty
176 */
177
178#define GET_RS232_BAUD	  13
179/**
180 * Params: none
181 *
182 * Answer: byte read from UART bit rate register (UBRR)
183 */
184
185
186/* data structure for each usb remote */
187struct igorplug {
188
189	/* usb */
190	struct usb_device *usbdev;
191	int devnum;
192
193	unsigned char *buf_in;
194	unsigned int len_in;
195	int in_space;
196	struct timeval last_time;
197
198	dma_addr_t dma_in;
199
200	/* lirc */
201	struct lirc_driver *d;
202
203	/* handle sending (init strings) */
204	int send_flags;
205};
206
207static int unregister_from_lirc(struct igorplug *ir)
208{
209	struct lirc_driver *d;
210	int devnum;
211
212	devnum = ir->devnum;
213	d = ir->d;
214
215	if (!d) {
216		dev_err(&ir->usbdev->dev,
217			"%s: called with NULL lirc driver struct!\n", __func__);
218		return -EINVAL;
219	}
220
221	dev_dbg(&ir->usbdev->dev, "calling lirc_unregister_driver\n");
222	lirc_unregister_driver(d->minor);
223
224	return devnum;
225}
226
227static int set_use_inc(void *data)
228{
229	struct igorplug *ir = data;
230
231	if (!ir) {
232		printk(DRIVER_NAME "[?]: set_use_inc called with no context\n");
233		return -EIO;
234	}
235
236	dev_dbg(&ir->usbdev->dev, "set use inc\n");
237
238	if (!ir->usbdev)
239		return -ENODEV;
240
241	return 0;
242}
243
244static void set_use_dec(void *data)
245{
246	struct igorplug *ir = data;
247
248	if (!ir) {
249		printk(DRIVER_NAME "[?]: set_use_dec called with no context\n");
250		return;
251	}
252
253	dev_dbg(&ir->usbdev->dev, "set use dec\n");
254}
255
256static void send_fragment(struct igorplug *ir, struct lirc_buffer *buf,
257			   int i, int max)
258{
259	int code;
260
261	/* MODE2: pulse/space (PULSE_BIT) in 1us units */
262	while (i < max) {
263		/* 1 Igor-tick = 85.333333 us */
264		code = (unsigned int)ir->buf_in[i] * 85 +
265			(unsigned int)ir->buf_in[i] / 3;
266		ir->last_time.tv_usec += code;
267		if (ir->in_space)
268			code |= PULSE_BIT;
269		lirc_buffer_write(buf, (unsigned char *)&code);
270		/* 1 chunk = CODE_LENGTH bytes */
271		ir->in_space ^= 1;
272		++i;
273	}
274}
275
276/**
277 * Called in user context.
278 * return 0 if data was added to the buffer and
279 * -ENODATA if none was available. This should add some number of bits
280 * evenly divisible by code_length to the buffer
281 */
282static int igorplugusb_remote_poll(void *data, struct lirc_buffer *buf)
283{
284	int ret;
285	struct igorplug *ir = (struct igorplug *)data;
286
287	if (!ir || !ir->usbdev)  /* Has the device been removed? */
288		return -ENODEV;
289
290	memset(ir->buf_in, 0, ir->len_in);
291
292	ret = usb_control_msg(ir->usbdev, usb_rcvctrlpipe(ir->usbdev, 0),
293			      GET_INFRACODE, USB_TYPE_VENDOR | USB_DIR_IN,
294			      0/* offset */, /*unused*/0,
295			      ir->buf_in, ir->len_in,
296			      /*timeout*/HZ * USB_CTRL_GET_TIMEOUT);
297	if (ret > 0) {
298		int code, timediff;
299		struct timeval now;
300
301		/* ACK packet has 1 byte --> ignore */
302		if (ret < DEVICE_HEADERLEN)
303			return -ENODATA;
304
305		dev_dbg(&ir->usbdev->dev, "Got %d bytes. Header: %*ph\n",
306			ret, 3, ir->buf_in);
307
308		do_gettimeofday(&now);
309		timediff = now.tv_sec - ir->last_time.tv_sec;
310		if (timediff + 1 > PULSE_MASK / 1000000)
311			timediff = PULSE_MASK;
312		else {
313			timediff *= 1000000;
314			timediff += now.tv_usec - ir->last_time.tv_usec;
315		}
316		ir->last_time.tv_sec = now.tv_sec;
317		ir->last_time.tv_usec = now.tv_usec;
318
319		/* create leading gap  */
320		code = timediff;
321		lirc_buffer_write(buf, (unsigned char *)&code);
322		ir->in_space = 1;   /* next comes a pulse */
323
324		if (ir->buf_in[2] == 0)
325			send_fragment(ir, buf, DEVICE_HEADERLEN, ret);
326		else {
327			dev_warn(&ir->usbdev->dev,
328				 "[%d]: Device buffer overrun.\n", ir->devnum);
329			/* HHHNNNNNNNNNNNOOOOOOOO H = header
330			      <---[2]--->         N = newer
331			   <---------ret--------> O = older */
332			ir->buf_in[2] %= ret - DEVICE_HEADERLEN; /* sanitize */
333			/* keep even-ness to not desync pulse/pause */
334			send_fragment(ir, buf, DEVICE_HEADERLEN +
335				      ir->buf_in[2] - (ir->buf_in[2] & 1), ret);
336			send_fragment(ir, buf, DEVICE_HEADERLEN,
337				      DEVICE_HEADERLEN + ir->buf_in[2]);
338		}
339
340		ret = usb_control_msg(
341		      ir->usbdev, usb_rcvctrlpipe(ir->usbdev, 0),
342		      SET_INFRABUFFER_EMPTY, USB_TYPE_VENDOR|USB_DIR_IN,
343		      /*unused*/0, /*unused*/0,
344		      /*dummy*/ir->buf_in, /*dummy*/ir->len_in,
345		      /*timeout*/HZ * USB_CTRL_GET_TIMEOUT);
346		if (ret < 0)
347			printk(DRIVER_NAME "[%d]: SET_INFRABUFFER_EMPTY: error %d\n",
348			       ir->devnum, ret);
349		return 0;
350	} else if (ret < 0)
351		printk(DRIVER_NAME "[%d]: GET_INFRACODE: error %d\n",
352			ir->devnum, ret);
353
354	return -ENODATA;
355}
356
357static int igorplugusb_remote_probe(struct usb_interface *intf,
358				    const struct usb_device_id *id)
359{
360	struct usb_device *dev;
361	struct usb_host_interface *idesc = NULL;
362	struct usb_endpoint_descriptor *ep;
363	struct igorplug *ir = NULL;
364	struct lirc_driver *driver = NULL;
365	int devnum, pipe, maxp;
366	char buf[63], name[128] = "";
367	int ret;
368
369	dev_dbg(&intf->dev, "%s: usb probe called.\n", __func__);
370
371	dev = interface_to_usbdev(intf);
372
373	idesc = intf->cur_altsetting;
374
375	if (idesc->desc.bNumEndpoints != 1)
376		return -ENODEV;
377
378	ep = &idesc->endpoint->desc;
379	if (((ep->bEndpointAddress & USB_ENDPOINT_DIR_MASK)
380	    != USB_DIR_IN)
381	    || (ep->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK)
382	    != USB_ENDPOINT_XFER_CONTROL)
383		return -ENODEV;
384
385	pipe = usb_rcvctrlpipe(dev, ep->bEndpointAddress);
386	devnum = dev->devnum;
387	maxp = usb_maxpacket(dev, pipe, usb_pipeout(pipe));
388
389	dev_dbg(&intf->dev, "%s: bytes_in_key=%zu maxp=%d\n",
390		__func__, CODE_LENGTH, maxp);
391
392	ir = devm_kzalloc(&intf->dev, sizeof(*ir), GFP_KERNEL);
393	if (!ir)
394		return -ENOMEM;
395
396	driver = devm_kzalloc(&intf->dev, sizeof(*driver), GFP_KERNEL);
397	if (!driver)
398		return -ENOMEM;
399
400	ir->buf_in = usb_alloc_coherent(dev, DEVICE_BUFLEN + DEVICE_HEADERLEN,
401					GFP_ATOMIC, &ir->dma_in);
402	if (!ir->buf_in)
403		return -ENOMEM;
404
405	strcpy(driver->name, DRIVER_NAME " ");
406	driver->minor = -1;
407	driver->code_length = CODE_LENGTH * 8; /* in bits */
408	driver->features = LIRC_CAN_REC_MODE2;
409	driver->data = ir;
410	driver->chunk_size = CODE_LENGTH;
411	driver->buffer_size = DEVICE_BUFLEN + ADDITIONAL_LIRC_BYTES;
412	driver->set_use_inc = &set_use_inc;
413	driver->set_use_dec = &set_use_dec;
414	driver->sample_rate = sample_rate;    /* per second */
415	driver->add_to_buf = &igorplugusb_remote_poll;
416	driver->dev = &intf->dev;
417	driver->owner = THIS_MODULE;
418
419	ret = lirc_register_driver(driver);
420	if (ret < 0) {
421		usb_free_coherent(dev, DEVICE_BUFLEN + DEVICE_HEADERLEN,
422			ir->buf_in, ir->dma_in);
423		return ret;
424	}
425
426	driver->minor = ret;
427	ir->d = driver;
428	ir->devnum = devnum;
429	ir->usbdev = dev;
430	ir->len_in = DEVICE_BUFLEN + DEVICE_HEADERLEN;
431	ir->in_space = 1; /* First mode2 event is a space. */
432	do_gettimeofday(&ir->last_time);
433
434	if (dev->descriptor.iManufacturer
435	    && usb_string(dev, dev->descriptor.iManufacturer,
436			  buf, sizeof(buf)) > 0)
437		strlcpy(name, buf, sizeof(name));
438	if (dev->descriptor.iProduct
439	    && usb_string(dev, dev->descriptor.iProduct, buf, sizeof(buf)) > 0)
440		snprintf(name + strlen(name), sizeof(name) - strlen(name),
441			 " %s", buf);
442	printk(DRIVER_NAME "[%d]: %s on usb%d:%d\n", devnum, name,
443	       dev->bus->busnum, devnum);
444
445	/* clear device buffer */
446	ret = usb_control_msg(ir->usbdev, usb_rcvctrlpipe(ir->usbdev, 0),
447		SET_INFRABUFFER_EMPTY, USB_TYPE_VENDOR|USB_DIR_IN,
448		/*unused*/0, /*unused*/0,
449		/*dummy*/ir->buf_in, /*dummy*/ir->len_in,
450		/*timeout*/HZ * USB_CTRL_GET_TIMEOUT);
451	if (ret < 0)
452		printk(DRIVER_NAME "[%d]: SET_INFRABUFFER_EMPTY: error %d\n",
453			devnum, ret);
454
455	usb_set_intfdata(intf, ir);
456	return 0;
457}
458
459static void igorplugusb_remote_disconnect(struct usb_interface *intf)
460{
461	struct usb_device *usbdev = interface_to_usbdev(intf);
462	struct igorplug *ir = usb_get_intfdata(intf);
463	struct device *dev = &intf->dev;
464	int devnum;
465
466	usb_set_intfdata(intf, NULL);
467
468	if (!ir || !ir->d)
469		return;
470
471	ir->usbdev = NULL;
472
473	usb_free_coherent(usbdev, ir->len_in, ir->buf_in, ir->dma_in);
474
475	devnum = unregister_from_lirc(ir);
476
477	dev_info(dev, DRIVER_NAME "[%d]: %s done\n", devnum, __func__);
478}
479
480static struct usb_device_id igorplugusb_remote_id_table[] = {
481	/* Igor Plug USB (Atmel's Manufact. ID) */
482	{ USB_DEVICE(0x03eb, 0x0002) },
483	/* Fit PC2 Infrared Adapter */
484	{ USB_DEVICE(0x03eb, 0x21fe) },
485
486	/* Terminating entry */
487	{ }
488};
489
490static struct usb_driver igorplugusb_remote_driver = {
491	.name =		DRIVER_NAME,
492	.probe =	igorplugusb_remote_probe,
493	.disconnect =	igorplugusb_remote_disconnect,
494	.id_table =	igorplugusb_remote_id_table
495};
496
497module_usb_driver(igorplugusb_remote_driver);
498
499#include <linux/vermagic.h>
500MODULE_INFO(vermagic, VERMAGIC_STRING);
501
502MODULE_DESCRIPTION(DRIVER_DESC);
503MODULE_AUTHOR(DRIVER_AUTHOR);
504MODULE_LICENSE("GPL");
505MODULE_DEVICE_TABLE(usb, igorplugusb_remote_id_table);
506
507module_param(sample_rate, int, S_IRUGO | S_IWUSR);
508MODULE_PARM_DESC(sample_rate, "Sampling rate in Hz (default: 100)");
509