diff options
author | Patrick Boettcher <devnull@localhost> | 2004-09-05 11:42:13 +0000 |
---|---|---|
committer | Patrick Boettcher <devnull@localhost> | 2004-09-05 11:42:13 +0000 |
commit | 0ea5aa58975517299dbb1688dbf879603291716a (patch) | |
tree | a4ebff0c51a159a17bb4c270a23c0d17bb7b4f62 /linux/drivers/media/dvb/dibusb/dvb-dibusb.c | |
parent | 7d6379ce8a3f77ec43a279591cbaa37eb052c8d6 (diff) | |
download | mediapointer-dvb-s2-0ea5aa58975517299dbb1688dbf879603291716a.tar.gz mediapointer-dvb-s2-0ea5aa58975517299dbb1688dbf879603291716a.tar.bz2 |
dvb-dibusb is substituting the vp7041 USB driver. it is basicly a i2c-to-usb bridge and does the demux stuff (i.e. passing the mpeg2-data to the sw-demux).
Diffstat (limited to 'linux/drivers/media/dvb/dibusb/dvb-dibusb.c')
-rw-r--r-- | linux/drivers/media/dvb/dibusb/dvb-dibusb.c | 715 |
1 files changed, 715 insertions, 0 deletions
diff --git a/linux/drivers/media/dvb/dibusb/dvb-dibusb.c b/linux/drivers/media/dvb/dibusb/dvb-dibusb.c new file mode 100644 index 000000000..392f801b5 --- /dev/null +++ b/linux/drivers/media/dvb/dibusb/dvb-dibusb.c @@ -0,0 +1,715 @@ +/* + * Driver for mobile USB Budget DVB-T devices based on reference + * design made by DiBcom (http://www.dibcom.fr/) + * + * USB 1.1 devices: + * + * Twinhan VisionPlus VisionDTV USB-Ter DVB-T Device (VP7041) + * http://www.twinhan.com/visiontv-2_4.htm + * + * CTS Portable (Chinese Television System) + * http://www.2cts.tv/ctsportable/ + * + * KWorld V-Stream XPERT DTV - DVB-T USB + * http://www.kworld.com.tw/asp/pindex.asp?id=4&pid=13 + * + * HAMA DVB-T USB device + * + * + * dvb-dibusb.c + * + * Copyright (C) 2004 Patrick Boettcher (patrick.boettcher@desy.de) + * + * based on GPL code from DiBcom, which has + * + * Copyright (C) 2004 Amaury Demol for DiBcom (ademol@dibcom.fr) + * + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation, version 2. + * + * Acknowledgements + * + * Amaury Demol (ademol@dibcom.fr) from DiBcom for providing specs and driver + * sources, on which this driver (and the dib3000mb frontend) are based. + * + * TODO + * - probing for i2c addresses, it is possible, that they have been changed + * by the vendor + * + */ + +#include <linux/config.h> +#include <linux/kernel.h> +#include <linux/usb.h> +#include <linux/firmware.h> +#include <linux/version.h> +#include <linux/moduleparam.h> +#include <linux/pci.h> + +#include "dmxdev.h" +#include "dvb_demux.h" +#include "dvb_filter.h" +#include "dvb_net.h" +#include "dvb_frontend.h" + +#include "dvb-dibusb.h" + +static int debug; +module_param(debug, int, 0x644); +MODULE_PARM_DESC(debug, "Turn on/off debugging (default:off)."); + +#define dprintk if (debug) printk +#define debug_dump(b,l) if (debug) {\ + int i; printk("%s: %d > ",__FUNCTION__,l); \ + for (i = 0; i < l; i++) printk("%02x ", b[i]); \ + printk("\n");\ +} + +/* Version information */ +#define DRIVER_VERSION "0.0" +#define DRIVER_DESC "DiBcom based USB Budget DVB-T device" +#define DRIVER_AUTHOR "Patrick Boettcher, patrick.boettcher@desy.de" + +/* USB Driver stuff */ + +/* table of devices that work with this driver */ +static struct usb_device_id dibusb_table [] = { + { USB_DEVICE(USB_TWINHAN_VENDOR_ID, USB_VP7041_PRODUCT_PREFW_ID) }, + { USB_DEVICE(USB_TWINHAN_VENDOR_ID, USB_VP7041_PRODUCT_ID) }, + { USB_DEVICE(USB_KWORLD_VENDOR_ID, USB_VSTREAM_PRODUCT_PREFW_ID) }, + { USB_DEVICE(USB_KWORLD_VENDOR_ID, USB_VSTREAM_PRODUCT_ID) }, + { USB_DEVICE(USB_DIBCOM_VENDOR_ID, USB_DIBCOM_PRODUCT_PREFW_ID) }, + { USB_DEVICE(USB_DIBCOM_VENDOR_ID, USB_DIBCOM_PRODUCT_ID) }, + { } /* Terminating entry */ +}; + +MODULE_DEVICE_TABLE (usb, dibusb_table); + +static int dibusb_readwrite_usb(struct usb_dibusb *dib, + u8 *wbuf, u16 wlen, u8 *rbuf, u16 rlen) +{ + int actlen,ret = -ENOMEM; + + if (wbuf == NULL || wlen == 0) + return -EINVAL; + +/* if (dib->disconnecting) + return -EINVAL;*/ + + if ((ret = down_interruptible(&dib->usb_sem))) + return ret; + + debug_dump(wbuf,wlen); + + ret = usb_bulk_msg(dib->udev,COMMAND_PIPE, + wbuf,wlen,&actlen,DIBUSB_I2C_TIMEOUT); + + if (ret) + err("bulk message failed: %d (%d/%d)",ret,wlen,actlen); + else + ret = actlen != wlen ? -1 : 0; + + /* an answer is expected */ + if (!ret && rbuf && rlen) { + ret = usb_bulk_msg(dib->udev,RESULT_PIPE,rbuf,rlen, + &actlen,DIBUSB_I2C_TIMEOUT); + + if (ret) + err("recv bulk message failed: %d",ret); + else { + /*dprintk("rlen: %d\n",rlen);*/ + debug_dump(rbuf,actlen); + } + } + + up(&dib->usb_sem); + return ret; +} + +static int dibusb_write_usb(struct usb_dibusb *dib, u8 *buf, u16 len) +{ + return dibusb_readwrite_usb(dib,buf,len,NULL,0); +} + +static int dibusb_i2c_msg(struct usb_dibusb *dib, u8 addr, + u8 *wbuf, u16 wlen, u8 *rbuf, u16 rlen) +{ + u8 sndbuf[wlen+4]; /* lead(1) devaddr,direction(1) addr(2) data(wlen) (len(2) (when reading)) */ + /* write only ? */ + int wo = (rbuf == NULL || rlen == 0), + len = 2 + wlen + (wo ? 0 : 2); + +/* dprintk("wo: %d, wlen: %d, len: %d\n",wo,wlen,len);*/ + + sndbuf[0] = wo ? DIBUSB_REQ_I2C_WRITE : DIBUSB_REQ_I2C_READ; + sndbuf[1] = (addr & 0xfe) | (wo ? 0 : 1); + + memcpy(&sndbuf[2],wbuf,wlen); + + if (!wo) { + sndbuf[wlen+2] = (rlen >> 8) & 0xff; + sndbuf[wlen+3] = rlen & 0xff; + } + + return dibusb_readwrite_usb(dib,sndbuf,len,rbuf,rlen); +} + +/* + * DVB stuff + */ + +static struct dibusb_pid * dibusb_get_free_pid(struct usb_dibusb *dib) +{ + int i; + unsigned long flags; + struct dibusb_pid *dpid = NULL; + + spin_lock_irqsave(&dib->pid_list_lock,flags); + for (i=0; i < DIBUSB_MAX_PIDS; i++) + if (!dib->pid_list[i].active) { + dpid = dib->pid_list + i; + dpid->active = 1; + break; + } + spin_unlock_irqrestore(&dib->pid_list_lock,flags); + return dpid; +} + +static int dibusb_start_xfer(struct usb_dibusb *dib) +{ + u8 b[4] = { + (DIB3000MB_REG_FIFO >> 8) & 0xff, + (DIB3000MB_REG_FIFO) & 0xff, + (DIB3000MB_FIFO_ACTIVATE >> 8) & 0xff, + (DIB3000MB_FIFO_ACTIVATE) & 0xff + }; + return dibusb_i2c_msg(dib,DIBUSB_DEMOD_I2C_ADDR_DEFAULT,b,4,NULL,0); +} + +static int dibusb_stop_xfer(struct usb_dibusb *dib) +{ + u8 b[4] = { + (DIB3000MB_REG_FIFO >> 8) & 0xff, + (DIB3000MB_REG_FIFO) & 0xff, + (DIB3000MB_FIFO_INHIBIT >> 8) & 0xff, + (DIB3000MB_FIFO_INHIBIT) & 0xff + }; + return dibusb_i2c_msg(dib,DIBUSB_DEMOD_I2C_ADDR_DEFAULT,b,4,NULL,0); +} + +static int dibusb_set_pid(struct dibusb_pid *dpid) +{ + u16 pid = dpid->pid | (dpid->active ? DIB3000MB_ACTIVATE_FILTERING : 0); + u8 b[4] = { + (dpid->reg >> 8) & 0xff, + (dpid->reg) & 0xff, + (pid >> 8) & 0xff, + (pid) & 0xff + }; + + return dibusb_i2c_msg(dpid->dib,DIBUSB_DEMOD_I2C_ADDR_DEFAULT,b,4,NULL,0); +} + +static void dibusb_urb_complete(struct urb *urb, struct pt_regs *ptregs) +{ + struct usb_dibusb *dib = urb->context; + + if (!dib->streaming) + return; + + if (urb->status == 0) { + dprintk("URB return len: %d\n",urb->actual_length); + if (urb->actual_length % 188) + dprintk("TS Packets: %d, %d\n", urb->actual_length/188,urb->actual_length % 188); + dvb_dmx_swfilter_packets(&dib->demux, (u8*) urb->transfer_buffer,urb->actual_length/188); + } + + if (dib->streaming) + usb_submit_urb(urb,GFP_KERNEL); +} + + +static int dibusb_start_feed(struct dvb_demux_feed *dvbdmxfeed) +{ +// struct dvb_demux *dvbdmx = dvbdmxfeed->demux; + struct usb_dibusb *dib = dvbdmxfeed->demux->priv; + struct dibusb_pid *dpid; + int ret = 0; + + dprintk("pid: 0x%04x, feedtype: %d\n", dvbdmxfeed->pid,dvbdmxfeed->type); + + if ((dpid = dibusb_get_free_pid(dib)) == NULL) { + err("no free pid in list."); + return -ENODEV; + } + dvbdmxfeed->priv = dpid; + dpid->pid = dvbdmxfeed->pid; + + dibusb_set_pid(dpid); + + if (0 == dib->feed_count++) { + usb_fill_bulk_urb( dib->buf_urb, dib->udev, DATA_PIPE, + dib->buffer, 8192, dibusb_urb_complete, dib); + dib->buf_urb->transfer_flags = 0; + dib->buf_urb->timeout = 0; + + if ((ret = usb_submit_urb(dib->buf_urb,GFP_KERNEL))) { + dibusb_stop_xfer(dib); + err("could not submit buffer urb."); + return ret; + } + + if ((ret = dibusb_start_xfer(dib))) + return ret; + + dib->streaming = 1; + } + return 0; +} + +static int dibusb_stop_feed(struct dvb_demux_feed *dvbdmxfeed) +{ + struct usb_dibusb *dib = dvbdmxfeed->demux->priv; + struct dibusb_pid *dpid = (struct dibusb_pid *) dvbdmxfeed->priv; + + dprintk("stopfeed pid: 0x%04x, feedtype: %d",dvbdmxfeed->pid, dvbdmxfeed->type); + + if (dpid == NULL) + err("channel in dmxfeed->priv was NULL"); + else { + dpid->active = 0; + dpid->pid = 0; + dibusb_set_pid(dpid); + } + + if (--dib->feed_count == 0) { + dib->streaming = 0; + usb_unlink_urb(dib->buf_urb); + dibusb_stop_xfer(dib); + } + return 0; +} + +/* + * firmware transfers + */ + +/* + * do not use this, just a workaround for a bug, + * which will never occur :). + */ +static int dibusb_interrupt_read_loop(struct usb_dibusb *dib) +{ + u8 b[1] = { DIBUSB_REQ_INTR_READ }; + return dibusb_write_usb(dib,b,1); +} + +/* + * TODO: a tasklet should run with a delay of 1/10 second + * and fill an appropriate event device ? + */ +static int dibusb_read_remote_control(struct usb_dibusb *dib) +{ + u8 b[1] = { DIBUSB_REQ_POLL_REMOTE }, rb[5]; + int ret; + if ((ret = dibusb_readwrite_usb(dib,b,1,rb,5))) + return ret; + + return 0; +} + +/* + * ioctl for the firmware + */ +static int dibusb_ioctl_cmd(struct usb_dibusb *dib, u8 cmd, u8 *param, int plen) +{ + u8 b[34]; + int size = plen > 32 ? 32 : plen; + b[0] = DIBUSB_REQ_SET_IOCTL; + b[1] = cmd; + memcpy(&b[2],param,size); + + return dibusb_write_usb(dib,b,2+size); +} + +/* + * ioctl for power control + */ +static int dibusb_hw_sleep(struct usb_dibusb *dib) +{ + u8 b[1] = { DIBUSB_IOCTL_POWER_SLEEP }; + return dibusb_ioctl_cmd(dib,DIBUSB_IOCTL_CMD_POWER_MODE, b,1); +} + +static int dibusb_hw_wakeup(struct usb_dibusb *dib) +{ + u8 b[1] = { DIBUSB_IOCTL_POWER_WAKEUP }; + return dibusb_ioctl_cmd(dib,DIBUSB_IOCTL_CMD_POWER_MODE, b,1); +} + +/* + * I2C + */ +static int dibusb_i2c_xfer(struct i2c_adapter *adap,struct i2c_msg msg[],int num) +{ + struct usb_dibusb *dib = i2c_get_adapdata(adap); + int i; + + if (down_interruptible(&dib->i2c_sem) < 0) + return -EAGAIN; + + for (i = 0; i < num; i++) { + /* write/read request */ + if (i+1 < num && (msg[i+1].flags & I2C_M_RD)) { + if (dibusb_i2c_msg(dib, msg[i].addr, msg[i].buf,msg[i].len, + msg[i+1].buf,msg[i+1].len) < 0) + break; + i++; + } else + if (dibusb_i2c_msg(dib, msg[i].addr, msg[i].buf,msg[i].len,NULL,0) < 0) + break; + } + + up(&dib->i2c_sem); + return i; +} + +static u32 dibusb_i2c_func(struct i2c_adapter *adapter) +{ + return I2C_FUNC_I2C; +} + +static int dibusb_i2c_client_register (struct i2c_client *i2c) +{ + struct usb_dibusb *dib = i2c_get_adapdata(i2c->adapter); + if (i2c->driver->command) + return i2c->driver->command(i2c,FE_REGISTER,dib->adapter); + return 0; +} + +static int dibusb_i2c_client_unregister (struct i2c_client *i2c) +{ + struct usb_dibusb *dib = i2c_get_adapdata(i2c->adapter); + if (i2c->driver->command) + return i2c->driver->command(i2c,FE_UNREGISTER,dib->adapter); + return 0; +} + +static struct i2c_algorithm dibusb_algo = { + .name = "DiBcom USB i2c algorithm", + .id = I2C_ALGO_BIT, + .master_xfer = dibusb_i2c_xfer, + .functionality = dibusb_i2c_func, +}; + +static int dibusb_dvb_init(struct usb_dibusb *dib) +{ + int ret; + +#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,6,4) + if ((ret = dvb_register_adapter(&dib->adapter, DRIVER_DESC)) < 0) { +#else + if ((ret = dvb_register_adapter(&dib->adapter, DRIVER_DESC , + THIS_MODULE)) < 0) { +#endif + dprintk("dvb_register_adapter failed: error %d", ret); + goto err; + } + + strncpy(dib->i2c_adap.name,dib->dibdev->name,I2C_NAME_SIZE); +#ifdef I2C_ADAP_CLASS_TV_DIGITAL + dib->i2c_adap.class = I2C_ADAP_CLASS_TV_DIGITAL, +#else + dib->i2c_adap.class = I2C_CLASS_TV_DIGITAL, +#endif + dib->i2c_adap.algo = &dibusb_algo; + dib->i2c_adap.algo_data = NULL; + dib->i2c_adap.id = I2C_ALGO_BIT; + dib->i2c_adap.client_register = dibusb_i2c_client_register, + dib->i2c_adap.client_unregister = dibusb_i2c_client_unregister, + + i2c_set_adapdata(&dib->i2c_adap, dib); + + if ((i2c_add_adapter(&dib->i2c_adap) < 0)) { + err("could not add i2c adapter"); + goto err_i2c; + } + + dib->demux.dmx.capabilities = DMX_TS_FILTERING | DMX_SECTION_FILTERING; + + dib->demux.priv = (void *)dib; + dib->demux.filternum = DIBUSB_MAX_PIDS; + dib->demux.feednum = DIBUSB_MAX_PIDS; + dib->demux.start_feed = dibusb_start_feed; + dib->demux.stop_feed = dibusb_stop_feed; + dib->demux.write_to_decoder = NULL; + if ((ret = dvb_dmx_init(&dib->demux)) < 0) { + err("dvb_dmx_init failed: error %d",ret); + goto err_dmx; + } + + dib->dmxdev.filternum = dib->demux.filternum; + dib->dmxdev.demux = &dib->demux.dmx; + dib->dmxdev.capabilities = 0; + if ((ret = dvb_dmxdev_init(&dib->dmxdev, dib->adapter)) < 0) { + err("dvb_dmxdev_init failed: error %d",ret); + goto err_dmx_dev; + } + + dvb_net_init(dib->adapter, &dib->dvb_net, &dib->demux.dmx); + + goto success; +err_dmx_dev: + dvb_dmx_release(&dib->demux); +err_dmx: + i2c_del_adapter(&dib->i2c_adap); +err_i2c: + dvb_unregister_adapter(dib->adapter); +err: + return ret; +success: + return 0; +} + +static int dibusb_dvb_exit(struct usb_dibusb *dib) +{ + dprintk("unregistering DVB part\n"); + dvb_net_release(&dib->dvb_net); + dib->demux.dmx.close(&dib->demux.dmx); + dvb_dmxdev_release(&dib->dmxdev); + dvb_dmx_release(&dib->demux); + i2c_del_adapter(&dib->i2c_adap); + dvb_unregister_adapter(dib->adapter); + + return 0; +} + +static int dibusb_exit(struct usb_dibusb *dib) +{ + usb_free_urb(dib->buf_urb); + pci_free_consistent(NULL,8192,dib->buffer,dib->dma_handle); + return 0; +} + +static int dibusb_init(struct usb_dibusb *dib) +{ + int ret,i; + sema_init(&dib->usb_sem, 1); + sema_init(&dib->i2c_sem, 1); + + /* + * when reloading the driver w/o replugging the device + * a timeout occures, this seems to help + */ + usb_clear_halt(dib->udev,COMMAND_PIPE); + usb_clear_halt(dib->udev,RESULT_PIPE); + usb_clear_halt(dib->udev,DATA_PIPE); + + /* dibusb_reset_cpu(dib); */ + + dib->buffer = pci_alloc_consistent(NULL,8192, &dib->dma_handle); + memset(dib->buffer,0,8192); + if (!(dib->buf_urb = usb_alloc_urb(0,GFP_KERNEL))) { + dibusb_exit(dib); + return -ENOMEM; + } + + for (i=0; i < DIBUSB_MAX_PIDS; i++) { + dib->pid_list[i].reg = i+DIB3000MB_REG_FIRST_PID; + dib->pid_list[i].pid = 0; + dib->pid_list[i].active = 0; + dib->pid_list[i].dib = dib; + } + + dib->streaming = 0; + dib->feed_count = 0; + + if ((ret = dibusb_dvb_init(dib))) { + dibusb_exit(dib); + return ret; + } + return 0; +} + +/* + * load a firmware packet to the device + */ +static int dibusb_writemem(struct usb_device *udev,u16 addr,u8 *data, u8 len) +{ + return usb_control_msg(udev, usb_sndctrlpipe(udev,0), + 0xa0, USB_TYPE_VENDOR, addr, 0x00, data, len, 5*HZ); +} + +static int dibusb_loadfirmware(struct usb_device *udev, + struct dibusb_device *dibdev) +{ + const struct firmware *fw = NULL; + u16 addr; + u8 *b,*p; + int ret = 0,i; + + for (i = 0; i < sizeof(valid_firmware_filenames); i++) { + if ((ret = request_firmware(&fw, valid_firmware_filenames[i], &udev->dev)) == 0) { + info("using firmware file (%s).",valid_firmware_filenames[i]); + break; + } + dprintk("tried to find '%s' firmware - unsuccessful. (%d)\n", + valid_firmware_filenames[i],ret); + } + + if (fw == NULL) { + err("did not find a valid firmware file."); + return -EINVAL; + } + p = kmalloc(fw->size,GFP_KERNEL); + if (p != NULL) { + u8 reset; + /* + * you cannot use the fw->data as buffer for + * usb_control_msg, a new buffer has to be + * created + */ + memcpy(p,fw->data,fw->size); + + /* stop the CPU */ + reset = 1; + if ((ret = dibusb_writemem(udev,DIBUSB_CPU_CSREG,&reset,1)) != 1) + err("could not stop the USB controller CPU."); + for(i = 0; p[i+3] == 0 && i < fw->size; ) { + b = (u8 *) &p[i]; + addr = *((u16 *) &b[1]); + + ret = dibusb_writemem(udev,addr,&b[4],b[0]); + + if (ret != b[0]) { + err("error while transferring firmware " + "(transferred size: %d, block size: %d)", + ret,b[1]); + ret = -EINVAL; + break; + } + i += 5 + b[0]; + } + /* restart the CPU */ + reset = 0; + if ((ret = dibusb_writemem(udev,DIBUSB_CPU_CSREG,&reset,1)) != 1) + err("could not restart the USB controller CPU."); + + kfree(p); + ret = 0; + } else { + ret = -ENOMEM; + } + release_firmware(fw); + + return ret; +} + +/* + * USB + */ +static int dibusb_probe(struct usb_interface *intf, + const struct usb_device_id *id) +{ + struct usb_device *udev = interface_to_usbdev(intf); + struct usb_dibusb *dib = NULL; + struct dibusb_device *dibdev = NULL; + + int ret = -ENOMEM,i,cold=0; + + for (i = 0; i < DIBUSB_SUPPORTED_DEVICES; i++) + if (dibusb_devices[i].cold_product_id == udev->descriptor.idProduct || + dibusb_devices[i].warm_product_id == udev->descriptor.idProduct) { + dibdev = &dibusb_devices[i]; + + cold = dibdev->cold_product_id == udev->descriptor.idProduct; + + if (cold) + info("found a '%s' in cold state, will try to load a firmware",dibdev->name); + else + info("found a '%s' in warm state.",dibdev->name); + } + + if (dibdev == NULL) { + err("something went very wrong, " + "unknown product ID: %.4x",udev->descriptor.idProduct); + return -ENODEV; + } + + if (cold) + ret = dibusb_loadfirmware(udev,dibdev); + else { + dib = kmalloc(sizeof(struct usb_dibusb),GFP_KERNEL); + if (dib == NULL) { + err("no memory"); + return ret; + } + memset(dib,0,sizeof(struct usb_dibusb)); + + dib->udev = udev; + dib->dibdev = dibdev; + + usb_set_intfdata(intf, dib); + + ret = dibusb_init(dib); + } + + if (ret == 0) + info("%s successfully initialized and connected.",dibdev->name); + else + info("%s error while loading driver (%d)",dibdev->name,ret); + return ret; +} + +static void dibusb_disconnect(struct usb_interface *intf) +{ + struct usb_dibusb *dib = usb_get_intfdata(intf); + usb_set_intfdata(intf,NULL); + const char *name = DRIVER_DESC; + + if (dib != NULL) { + name = dib->dibdev->name; + dibusb_dvb_exit(dib); + dibusb_exit(dib); + kfree(dib); + } + info("%s successfully deinitialized and disconnected.",name); + +} + +/* usb specific object needed to register this driver with the usb subsystem */ +static struct usb_driver dibusb_driver = { + .owner = THIS_MODULE, + .name = "dvb_dibusb", + .probe = dibusb_probe, + .disconnect = dibusb_disconnect, + .id_table = dibusb_table, +}; + +/* module stuff */ +static int __init usb_dibusb_init(void) +{ + int result; + + if ((result = usb_register(&dibusb_driver))) { + err("usb_register failed. Error number %d",result); + return result; + } + + return 0; +} + +static void __exit usb_dibusb_exit(void) +{ + /* deregister this driver from the USB subsystem */ + usb_deregister(&dibusb_driver); +} + +module_init (usb_dibusb_init); +module_exit (usb_dibusb_exit); + +MODULE_AUTHOR(DRIVER_AUTHOR); +MODULE_DESCRIPTION(DRIVER_DESC); +MODULE_LICENSE("GPL"); |