|
@@ -0,0 +1,914 @@
|
|
|
+/*
|
|
|
+ * Hardware event => input event mapping:
|
|
|
+ *
|
|
|
+ *
|
|
|
+ *
|
|
|
+ input.h:#define BTN_TOOL_PEN 0x140 black
|
|
|
+ input.h:#define BTN_TOOL_RUBBER 0x141 blue
|
|
|
+ input.h:#define BTN_TOOL_BRUSH 0x142 green
|
|
|
+ input.h:#define BTN_TOOL_PENCIL 0x143 red
|
|
|
+ input.h:#define BTN_TOOL_AIRBRUSH 0x144 eraser
|
|
|
+ input.h:#define BTN_TOOL_FINGER 0x145 small eraser
|
|
|
+ input.h:#define BTN_TOOL_MOUSE 0x146 mimio interactive
|
|
|
+ input.h:#define BTN_TOOL_LENS 0x147 mimio interactive but1
|
|
|
+ input.h:#define LOCALBTN_TOOL_EXTRA1 0x14a mimio interactive but2 == BTN_TOUCH
|
|
|
+ input.h:#define LOCALBTN_TOOL_EXTRA2 0x14b mimio extra pens (orange, brown, yellow, purple) == BTN_STYLUS
|
|
|
+ input.h:#define LOCALBTN_TOOL_EXTRA3 0x14c unused == BTN_STYLUS2
|
|
|
+ input.h:#define BTN_TOOL_DOUBLETAP 0x14d unused
|
|
|
+ input.h:#define BTN_TOOL_TRIPLETAP 0x14e unused
|
|
|
+ *
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_K) => EV_KEY BIT(BTN_TOOL_PEN)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_B) => EV_KEY BIT(BTN_TOOL_RUBBER)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_G) => EV_KEY BIT(BTN_TOOL_BRUSH)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_R) => EV_KEY BIT(BTN_TOOL_PENCIL)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_E) => EV_KEY BIT(BTN_TOOL_AIRBRUSH)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_ES) => EV_KEY BIT(BTN_TOOL_FINGER)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_I) => EV_KEY BIT(BTN_TOOL_MOUSE)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_IL) => EV_KEY BIT(BTN_TOOL_LENS)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_IR) => EV_KEY BIT(BTN_TOOL_DOUBLETAP)
|
|
|
+ * MIMIO_EV_PENDOWN(MIMIO_PEN_EX) => EV_KEY BIT(BTN_TOOL_TRIPLETAP)
|
|
|
+ * MIMIO_EV_PENDATA => EV_ABS BIT(ABS_X), BIT(ABS_Y)
|
|
|
+ * MIMIO_EV_MEMRESET => EV_KEY BIT(BTN_0)
|
|
|
+ * MIMIO_EV_ACC(ACC_NEWPAGE) => EV_KEY BIT(BTN_1)
|
|
|
+ * MIMIO_EV_ACC(ACC_TAGPAGE) => EV_KEY BIT(BTN_2)
|
|
|
+ * MIMIO_EV_ACC(ACC_PRINTPAGE) => EV_KEY BIT(BTN_3)
|
|
|
+ * MIMIO_EV_ACC(ACC_MAXIMIZE) => EV_KEY BIT(BTN_4)
|
|
|
+ * MIMIO_EV_ACC(ACC_FINDCTLPNL) => EV_KEY BIT(BTN_5)
|
|
|
+ *
|
|
|
+ *
|
|
|
+ * open issues:
|
|
|
+ * - cold-load of data captured when mimio in standalone mode not yet
|
|
|
+ * supported; need to snoop Win32 box to see datastream for this.
|
|
|
+ * - mimio mouse not yet supported; need to snoop Win32 box to see the
|
|
|
+ * datastream for this.
|
|
|
+ */
|
|
|
+#include <linux/kernel.h>
|
|
|
+#include <linux/init.h>
|
|
|
+#include <linux/slab.h>
|
|
|
+#include <linux/spinlock.h>
|
|
|
+#include <linux/input.h>
|
|
|
+#include <linux/usb.h>
|
|
|
+
|
|
|
+#define DRIVER_VERSION "v0.031"
|
|
|
+#define DRIVER_AUTHOR "mwilder@cs.nmsu.edu"
|
|
|
+#define DRIVER_DESC "USB mimio-xi driver"
|
|
|
+
|
|
|
+enum {UPVALUE, DOWNVALUE, MOVEVALUE};
|
|
|
+
|
|
|
+#define MIMIO_XRANGE_MAX 9600
|
|
|
+#define MIMIO_YRANGE_MAX 4800
|
|
|
+
|
|
|
+#define LOCALBTN_TOOL_EXTRA1 BTN_TOUCH
|
|
|
+#define LOCALBTN_TOOL_EXTRA2 BTN_STYLUS
|
|
|
+#define LOCALBTN_TOOL_EXTRA3 BTN_STYLUS2
|
|
|
+
|
|
|
+#define MIMIO_VENDOR_ID 0x08d3
|
|
|
+#define MIMIO_PRODUCT_ID 0x0001
|
|
|
+#define MIMIO_MAXPAYLOAD (8)
|
|
|
+#define MIMIO_MAXNAMELEN (64)
|
|
|
+#define MIMIO_TXWAIT (1)
|
|
|
+#define MIMIO_TXDONE (2)
|
|
|
+
|
|
|
+#define MIMIO_EV_PENDOWN (0x22)
|
|
|
+#define MIMIO_EV_PENDATA (0x24)
|
|
|
+#define MIMIO_EV_PENUP (0x51)
|
|
|
+#define MIMIO_EV_MEMRESET (0x45)
|
|
|
+#define MIMIO_EV_ACC (0xb2)
|
|
|
+
|
|
|
+#define MIMIO_PEN_K (1) /* black pen */
|
|
|
+#define MIMIO_PEN_B (2) /* blue pen */
|
|
|
+#define MIMIO_PEN_G (3) /* green pen */
|
|
|
+#define MIMIO_PEN_R (4) /* red pen */
|
|
|
+/* 5, 6, 7, 8 are extra pens */
|
|
|
+#define MIMIO_PEN_E (9) /* big eraser */
|
|
|
+#define MIMIO_PEN_ES (10) /* lil eraser */
|
|
|
+#define MIMIO_PENJUMP_START (10)
|
|
|
+#define MIMIO_PENJUMP (6)
|
|
|
+#define MIMIO_PEN_I (17) /* mimio interactive */
|
|
|
+#define MIMIO_PEN_IL (18) /* mimio interactive button 1 */
|
|
|
+#define MIMIO_PEN_IR (19) /* mimio interactive button 2 */
|
|
|
+
|
|
|
+#define MIMIO_PEN_MAX (MIMIO_PEN_IR)
|
|
|
+
|
|
|
+#define ACC_DONE (0)
|
|
|
+#define ACC_NEWPAGE (1)
|
|
|
+#define ACC_TAGPAGE (2)
|
|
|
+#define ACC_PRINTPAGE (4)
|
|
|
+#define ACC_MAXIMIZE (8)
|
|
|
+#define ACC_FINDCTLPNL (16)
|
|
|
+
|
|
|
+#define isvalidtxsize(n) ((n) > 0 && (n) <= MIMIO_MAXPAYLOAD)
|
|
|
+
|
|
|
+
|
|
|
+struct pktbuf {
|
|
|
+ unsigned char instr;
|
|
|
+ unsigned char buf[16];
|
|
|
+ unsigned char *p;
|
|
|
+ unsigned char *q;
|
|
|
+};
|
|
|
+
|
|
|
+struct usbintendpt {
|
|
|
+ dma_addr_t dma;
|
|
|
+ struct urb *urb;
|
|
|
+ unsigned char *buf;
|
|
|
+ struct usb_endpoint_descriptor *desc;
|
|
|
+};
|
|
|
+
|
|
|
+struct mimio {
|
|
|
+ struct input_dev *idev;
|
|
|
+ struct usb_device *udev;
|
|
|
+ struct usb_interface *uifc;
|
|
|
+ int open;
|
|
|
+ int present;
|
|
|
+ int greeted;
|
|
|
+ int txflags;
|
|
|
+ char phys[MIMIO_MAXNAMELEN];
|
|
|
+ struct usbintendpt in;
|
|
|
+ struct usbintendpt out;
|
|
|
+ struct pktbuf pktbuf;
|
|
|
+ unsigned char minor;
|
|
|
+ wait_queue_head_t waitq;
|
|
|
+ spinlock_t txlock;
|
|
|
+ void (*rxhandler)(struct mimio *, unsigned char *, unsigned int);
|
|
|
+ int last_pen_down;
|
|
|
+};
|
|
|
+
|
|
|
+static void mimio_close(struct input_dev *);
|
|
|
+static void mimio_dealloc(struct mimio *);
|
|
|
+static void mimio_disconnect(struct usb_interface *);
|
|
|
+static int mimio_greet(struct mimio *);
|
|
|
+static void mimio_irq_in(struct urb *);
|
|
|
+static void mimio_irq_out(struct urb *);
|
|
|
+static int mimio_open(struct input_dev *);
|
|
|
+static int mimio_probe(struct usb_interface *, const struct usb_device_id *);
|
|
|
+static void mimio_rx_handler(struct mimio *, unsigned char *, unsigned int);
|
|
|
+static int mimio_tx(struct mimio *, const char *, int);
|
|
|
+
|
|
|
+static char mimio_name[] = "VirtualInk mimio-Xi";
|
|
|
+static struct usb_device_id mimio_table [] = {
|
|
|
+ { USB_DEVICE(MIMIO_VENDOR_ID, MIMIO_PRODUCT_ID) },
|
|
|
+ { USB_DEVICE(0x0525, 0xa4a0) }, /* gadget zero firmware */
|
|
|
+ { }
|
|
|
+};
|
|
|
+
|
|
|
+MODULE_DEVICE_TABLE(usb, mimio_table);
|
|
|
+
|
|
|
+static struct usb_driver mimio_driver = {
|
|
|
+ .name = "mimio",
|
|
|
+ .probe = mimio_probe,
|
|
|
+ .disconnect = mimio_disconnect,
|
|
|
+ .id_table = mimio_table,
|
|
|
+};
|
|
|
+
|
|
|
+static DECLARE_MUTEX(disconnect_sem);
|
|
|
+
|
|
|
+static void mimio_close(struct input_dev *idev)
|
|
|
+{
|
|
|
+ struct mimio *mimio;
|
|
|
+
|
|
|
+ mimio = input_get_drvdata(idev);
|
|
|
+ if (!mimio) {
|
|
|
+ dev_err(&idev->dev, "null mimio attached to input device\n");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (mimio->open <= 0)
|
|
|
+ dev_err(&idev->dev, "mimio not open.\n");
|
|
|
+ else
|
|
|
+ mimio->open--;
|
|
|
+
|
|
|
+ if (mimio->present == 0 && mimio->open == 0)
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+}
|
|
|
+
|
|
|
+static void mimio_dealloc(struct mimio *mimio)
|
|
|
+{
|
|
|
+ if (mimio == NULL)
|
|
|
+ return;
|
|
|
+
|
|
|
+ usb_kill_urb(mimio->in.urb);
|
|
|
+
|
|
|
+ usb_kill_urb(mimio->out.urb);
|
|
|
+
|
|
|
+ if (mimio->idev) {
|
|
|
+ input_unregister_device(mimio->idev);
|
|
|
+ if (mimio->idev->grab)
|
|
|
+ input_close_device(mimio->idev->grab);
|
|
|
+ else
|
|
|
+ dev_dbg(&mimio->idev->dev, "mimio->idev->grab == NULL"
|
|
|
+ " -- didn't call input_close_device\n");
|
|
|
+ }
|
|
|
+
|
|
|
+ usb_free_urb(mimio->in.urb);
|
|
|
+
|
|
|
+ usb_free_urb(mimio->out.urb);
|
|
|
+
|
|
|
+ if (mimio->in.buf) {
|
|
|
+ usb_buffer_free(mimio->udev, MIMIO_MAXPAYLOAD, mimio->in.buf,
|
|
|
+ mimio->in.dma);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (mimio->out.buf)
|
|
|
+ usb_buffer_free(mimio->udev, MIMIO_MAXPAYLOAD, mimio->out.buf,
|
|
|
+ mimio->out.dma);
|
|
|
+
|
|
|
+ if (mimio->idev)
|
|
|
+ input_free_device(mimio->idev);
|
|
|
+
|
|
|
+ kfree(mimio);
|
|
|
+}
|
|
|
+
|
|
|
+static void mimio_disconnect(struct usb_interface *ifc)
|
|
|
+{
|
|
|
+ struct mimio *mimio;
|
|
|
+
|
|
|
+ down(&disconnect_sem);
|
|
|
+
|
|
|
+ mimio = usb_get_intfdata(ifc);
|
|
|
+ usb_set_intfdata(ifc, NULL);
|
|
|
+ dev_dbg(&mimio->idev->dev, "disconnect\n");
|
|
|
+
|
|
|
+ if (mimio) {
|
|
|
+ mimio->present = 0;
|
|
|
+
|
|
|
+ if (mimio->open <= 0)
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ }
|
|
|
+
|
|
|
+ up(&disconnect_sem);
|
|
|
+}
|
|
|
+
|
|
|
+static int mimio_greet(struct mimio *mimio)
|
|
|
+{
|
|
|
+ const struct grtpkt {
|
|
|
+ int nbytes;
|
|
|
+ unsigned delay;
|
|
|
+ char data[8];
|
|
|
+ } grtpkts[] = {
|
|
|
+ { 3, 0, { 0x11, 0x55, 0x44, 0x00, 0x00, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x53, 0x55, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x43, 0x55, 0x00, 0x00, 0x16, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x33, 0x55, 0x00, 0x00, 0x66, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x13, 0x00, 0x5e, 0x02, 0x4f, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x13, 0x00, 0x04, 0x03, 0x14, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 2, { 0x13, 0x00, 0x00, 0x04, 0x17, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x13, 0x00, 0x0d, 0x08, 0x16, 0x00, 0x00, 0x00 } },
|
|
|
+ { 5, 0, { 0x13, 0x00, 0x4d, 0x01, 0x5f, 0x00, 0x00, 0x00 } },
|
|
|
+ { 3, 0, { 0xf1, 0x55, 0xa4, 0x00, 0x00, 0x00, 0x00, 0x00 } },
|
|
|
+ { 7, 2, { 0x52, 0x55, 0x00, 0x07, 0x31, 0x55, 0x64, 0x00 } },
|
|
|
+ { 0, 0, { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } },
|
|
|
+ };
|
|
|
+ int rslt;
|
|
|
+ const struct grtpkt *pkt;
|
|
|
+
|
|
|
+ for (pkt = grtpkts; pkt->nbytes; pkt++) {
|
|
|
+ rslt = mimio_tx(mimio, pkt->data, pkt->nbytes);
|
|
|
+ if (rslt)
|
|
|
+ return rslt;
|
|
|
+ if (pkt->delay)
|
|
|
+ msleep(pkt->delay);
|
|
|
+ }
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static void mimio_irq_in(struct urb *urb)
|
|
|
+{
|
|
|
+ int rslt;
|
|
|
+ char *data;
|
|
|
+ const char *reason = "going down";
|
|
|
+ struct mimio *mimio;
|
|
|
+
|
|
|
+ mimio = urb->context;
|
|
|
+
|
|
|
+ if (mimio == NULL)
|
|
|
+ /* paranoia */
|
|
|
+ return;
|
|
|
+
|
|
|
+ switch (urb->status) {
|
|
|
+ case 0:
|
|
|
+ /* success */
|
|
|
+ break;
|
|
|
+ case -ETIMEDOUT:
|
|
|
+ reason = "timeout -- unplugged?";
|
|
|
+ case -ECONNRESET:
|
|
|
+ case -ENOENT:
|
|
|
+ case -ESHUTDOWN:
|
|
|
+ dev_dbg(&mimio->idev->dev, "%s.\n", reason);
|
|
|
+ return;
|
|
|
+ default:
|
|
|
+ dev_dbg(&mimio->idev->dev, "unknown urb-status: %d.\n",
|
|
|
+ urb->status);
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+ data = mimio->in.buf;
|
|
|
+
|
|
|
+ if (mimio->rxhandler)
|
|
|
+ mimio->rxhandler(mimio, data, urb->actual_length);
|
|
|
+exit:
|
|
|
+ /*
|
|
|
+ * Keep listening to device on same urb.
|
|
|
+ */
|
|
|
+ rslt = usb_submit_urb(urb, GFP_ATOMIC);
|
|
|
+ if (rslt)
|
|
|
+ dev_err(&mimio->idev->dev, "usb_submit_urb failure: %d.\n",
|
|
|
+ rslt);
|
|
|
+}
|
|
|
+
|
|
|
+static void mimio_irq_out(struct urb *urb)
|
|
|
+{
|
|
|
+ unsigned long flags;
|
|
|
+ struct mimio *mimio;
|
|
|
+
|
|
|
+ mimio = urb->context;
|
|
|
+
|
|
|
+ if (urb->status)
|
|
|
+ dev_dbg(&mimio->idev->dev, "urb-status: %d.\n", urb->status);
|
|
|
+
|
|
|
+ spin_lock_irqsave(&mimio->txlock, flags);
|
|
|
+ mimio->txflags |= MIMIO_TXDONE;
|
|
|
+ spin_unlock_irqrestore(&mimio->txlock, flags);
|
|
|
+ wmb();
|
|
|
+ wake_up(&mimio->waitq);
|
|
|
+}
|
|
|
+
|
|
|
+static int mimio_open(struct input_dev *idev)
|
|
|
+{
|
|
|
+ int rslt;
|
|
|
+ struct mimio *mimio;
|
|
|
+
|
|
|
+ rslt = 0;
|
|
|
+ down(&disconnect_sem);
|
|
|
+ mimio = input_get_drvdata(idev);
|
|
|
+ dev_dbg(&idev->dev, "mimio_open\n");
|
|
|
+
|
|
|
+ if (mimio == NULL) {
|
|
|
+ dev_err(&idev->dev, "null mimio.\n");
|
|
|
+ rslt = -ENODEV;
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (mimio->open++)
|
|
|
+ goto exit;
|
|
|
+
|
|
|
+ if (mimio->present && !mimio->greeted) {
|
|
|
+ struct urb *urb = mimio->in.urb;
|
|
|
+ mimio->in.urb->dev = mimio->udev;
|
|
|
+ rslt = usb_submit_urb(mimio->in.urb, GFP_KERNEL);
|
|
|
+ if (rslt) {
|
|
|
+ dev_err(&idev->dev, "usb_submit_urb failure "
|
|
|
+ "(res = %d: %s). Not greeting.\n",
|
|
|
+ rslt,
|
|
|
+ (!urb ? "urb is NULL" :
|
|
|
+ (urb->hcpriv ? "urb->hcpriv is non-NULL" :
|
|
|
+ (!urb->complete ? "urb is not complete" :
|
|
|
+ (urb->number_of_packets <= 0 ? "urb has no packets" :
|
|
|
+ (urb->interval <= 0 ? "urb interval too small" :
|
|
|
+ "urb interval too large or some other error"))))));
|
|
|
+ rslt = -EIO;
|
|
|
+ goto exit;
|
|
|
+ }
|
|
|
+ rslt = mimio_greet(mimio);
|
|
|
+ if (rslt == 0) {
|
|
|
+ dev_dbg(&idev->dev, "Mimio greeted OK.\n");
|
|
|
+ mimio->greeted = 1;
|
|
|
+ } else {
|
|
|
+ dev_dbg(&idev->dev, "Mimio greet Failure (%d)\n",
|
|
|
+ rslt);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+exit:
|
|
|
+ up(&disconnect_sem);
|
|
|
+ return rslt;
|
|
|
+}
|
|
|
+
|
|
|
+static int mimio_probe(struct usb_interface *ifc,
|
|
|
+ const struct usb_device_id *id)
|
|
|
+{
|
|
|
+ char path[64];
|
|
|
+ int pipe, maxp;
|
|
|
+ struct mimio *mimio;
|
|
|
+ struct usb_device *udev;
|
|
|
+ struct usb_host_interface *hostifc;
|
|
|
+ struct input_dev *input_dev;
|
|
|
+ int res = 0;
|
|
|
+ int i;
|
|
|
+
|
|
|
+ udev = interface_to_usbdev(ifc);
|
|
|
+
|
|
|
+ mimio = kzalloc(sizeof(struct mimio), GFP_KERNEL);
|
|
|
+ if (!mimio)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ input_dev = input_allocate_device();
|
|
|
+ if (!input_dev) {
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ mimio->uifc = ifc;
|
|
|
+ mimio->udev = udev;
|
|
|
+ mimio->pktbuf.p = mimio->pktbuf.buf;
|
|
|
+ mimio->pktbuf.q = mimio->pktbuf.buf;
|
|
|
+ /* init_input_dev(mimio->idev); */
|
|
|
+ mimio->idev = input_dev;
|
|
|
+ init_waitqueue_head(&mimio->waitq);
|
|
|
+ spin_lock_init(&mimio->txlock);
|
|
|
+ hostifc = ifc->cur_altsetting;
|
|
|
+
|
|
|
+ if (hostifc->desc.bNumEndpoints != 2) {
|
|
|
+ dev_err(&udev->dev, "Unexpected endpoint count: %d.\n",
|
|
|
+ hostifc->desc.bNumEndpoints);
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -ENODEV;
|
|
|
+ }
|
|
|
+
|
|
|
+ mimio->in.desc = &(hostifc->endpoint[0].desc);
|
|
|
+ mimio->out.desc = &(hostifc->endpoint[1].desc);
|
|
|
+
|
|
|
+ mimio->in.buf = usb_buffer_alloc(udev, MIMIO_MAXPAYLOAD, GFP_KERNEL,
|
|
|
+ &mimio->in.dma);
|
|
|
+ mimio->out.buf = usb_buffer_alloc(udev, MIMIO_MAXPAYLOAD, GFP_KERNEL,
|
|
|
+ &mimio->out.dma);
|
|
|
+
|
|
|
+ if (mimio->in.buf == NULL || mimio->out.buf == NULL) {
|
|
|
+ dev_err(&udev->dev, "usb_buffer_alloc failure.\n");
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ mimio->in.urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
+ mimio->out.urb = usb_alloc_urb(0, GFP_KERNEL);
|
|
|
+
|
|
|
+ if (mimio->in.urb == NULL || mimio->out.urb == NULL) {
|
|
|
+ dev_err(&udev->dev, "usb_alloc_urb failure.\n");
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Build the input urb.
|
|
|
+ */
|
|
|
+ pipe = usb_rcvintpipe(udev, mimio->in.desc->bEndpointAddress);
|
|
|
+ maxp = usb_maxpacket(udev, pipe, usb_pipeout(pipe));
|
|
|
+ if (maxp > MIMIO_MAXPAYLOAD)
|
|
|
+ maxp = MIMIO_MAXPAYLOAD;
|
|
|
+ usb_fill_int_urb(mimio->in.urb, udev, pipe, mimio->in.buf, maxp,
|
|
|
+ mimio_irq_in, mimio, mimio->in.desc->bInterval);
|
|
|
+ mimio->in.urb->transfer_dma = mimio->in.dma;
|
|
|
+ mimio->in.urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Build the output urb.
|
|
|
+ */
|
|
|
+ pipe = usb_sndintpipe(udev, mimio->out.desc->bEndpointAddress);
|
|
|
+ maxp = usb_maxpacket(udev, pipe, usb_pipeout(pipe));
|
|
|
+ if (maxp > MIMIO_MAXPAYLOAD)
|
|
|
+ maxp = MIMIO_MAXPAYLOAD;
|
|
|
+ usb_fill_int_urb(mimio->out.urb, udev, pipe, mimio->out.buf, maxp,
|
|
|
+ mimio_irq_out, mimio, mimio->out.desc->bInterval);
|
|
|
+ mimio->out.urb->transfer_dma = mimio->out.dma;
|
|
|
+ mimio->out.urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Build input device info
|
|
|
+ */
|
|
|
+ usb_make_path(udev, path, 64);
|
|
|
+ snprintf(mimio->phys, MIMIO_MAXNAMELEN, "%s/input0", path);
|
|
|
+ input_set_drvdata(input_dev, mimio);
|
|
|
+ /* input_dev->dev = &ifc->dev; */
|
|
|
+ input_dev->open = mimio_open;
|
|
|
+ input_dev->close = mimio_close;
|
|
|
+ input_dev->name = mimio_name;
|
|
|
+ input_dev->phys = mimio->phys;
|
|
|
+ input_dev->dev.parent = &ifc->dev;
|
|
|
+
|
|
|
+ input_dev->id.bustype = BUS_USB;
|
|
|
+ input_dev->id.vendor = le16_to_cpu(udev->descriptor.idVendor);
|
|
|
+ input_dev->id.product = le16_to_cpu(udev->descriptor.idProduct);
|
|
|
+ input_dev->id.version = le16_to_cpu(udev->descriptor.bcdDevice);
|
|
|
+
|
|
|
+ input_dev->evbit[0] |= BIT(EV_KEY) | BIT(EV_ABS);
|
|
|
+ for (i = BTN_TOOL_PEN; i <= LOCALBTN_TOOL_EXTRA2; ++i)
|
|
|
+ set_bit(i, input_dev->keybit);
|
|
|
+
|
|
|
+ input_dev->keybit[BIT_WORD(BTN_MISC)] |= BIT_MASK(BTN_0) |
|
|
|
+ BIT_MASK(BTN_1) |
|
|
|
+ BIT_MASK(BTN_2) |
|
|
|
+ BIT_MASK(BTN_3) |
|
|
|
+ BIT_MASK(BTN_4) |
|
|
|
+ BIT_MASK(BTN_5);
|
|
|
+ /* input_dev->keybit[BTN_MOUSE] |= BIT(BTN_LEFT); */
|
|
|
+ input_dev->absbit[0] |= BIT_MASK(ABS_X) | BIT_MASK(ABS_Y);
|
|
|
+ input_set_abs_params(input_dev, ABS_X, 0, MIMIO_XRANGE_MAX, 0, 0);
|
|
|
+ input_set_abs_params(input_dev, ABS_Y, 0, MIMIO_YRANGE_MAX, 0, 0);
|
|
|
+ input_dev->absbit[BIT_WORD(ABS_MISC)] |= BIT_MASK(ABS_MISC);
|
|
|
+
|
|
|
+#if 0
|
|
|
+ input_dev->absmin[ABS_X] = 0;
|
|
|
+ input_dev->absmin[ABS_Y] = 0;
|
|
|
+ input_dev->absmax[ABS_X] = 9600;
|
|
|
+ input_dev->absmax[ABS_Y] = 4800;
|
|
|
+ input_dev->absfuzz[ABS_X] = 0;
|
|
|
+ input_dev->absfuzz[ABS_Y] = 0;
|
|
|
+ input_dev->absflat[ABS_X] = 0;
|
|
|
+ input_dev->absflat[ABS_Y] = 0;
|
|
|
+#endif
|
|
|
+
|
|
|
+#if 0
|
|
|
+ /* this will just reduce the precision */
|
|
|
+ input_dev->absfuzz[ABS_X] = 8; /* experimental; may need to change */
|
|
|
+ input_dev->absfuzz[ABS_Y] = 8; /* experimental; may need to change */
|
|
|
+#endif
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Register the input device.
|
|
|
+ */
|
|
|
+ res = input_register_device(mimio->idev);
|
|
|
+ if (res) {
|
|
|
+ dev_err(&udev->dev, "input_register_device failure (%d)\n",
|
|
|
+ res);
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -EIO;
|
|
|
+ }
|
|
|
+ dev_dbg(&mimio->idev->dev, "input: %s on %s (res = %d).\n",
|
|
|
+ input_dev->name, input_dev->phys, res);
|
|
|
+
|
|
|
+ usb_set_intfdata(ifc, mimio);
|
|
|
+ mimio->present = 1;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Submit the input urb to the usb subsystem.
|
|
|
+ */
|
|
|
+ mimio->in.urb->dev = mimio->udev;
|
|
|
+ res = usb_submit_urb(mimio->in.urb, GFP_KERNEL);
|
|
|
+ if (res) {
|
|
|
+ dev_err(&mimio->idev->dev, "usb_submit_urb failure (%d)\n",
|
|
|
+ res);
|
|
|
+ mimio_dealloc(mimio);
|
|
|
+ return -EIO;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Attempt to greet the mimio after giving
|
|
|
+ * it some post-init settling time.
|
|
|
+ *
|
|
|
+ * note: sometimes this sleep interval isn't
|
|
|
+ * long enough to permit the device to re-init
|
|
|
+ * after a hot-swap; maybe need to bump it up.
|
|
|
+ *
|
|
|
+ * As it is, this probably breaks module unloading support!
|
|
|
+ */
|
|
|
+ msleep(1024);
|
|
|
+
|
|
|
+ res = mimio_greet(mimio);
|
|
|
+ if (res == 0) {
|
|
|
+ dev_dbg(&mimio->idev->dev, "Mimio greeted OK.\n");
|
|
|
+ mimio->greeted = 1;
|
|
|
+ mimio->rxhandler = mimio_rx_handler;
|
|
|
+ } else {
|
|
|
+ dev_dbg(&mimio->idev->dev, "Mimio greet Failure (%d)\n", res);
|
|
|
+ }
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static int handle_mimio_rx_penupdown(struct mimio *mimio,
|
|
|
+ int down,
|
|
|
+ const char *const instr[],
|
|
|
+ const int instr_ofst[])
|
|
|
+{
|
|
|
+ int penid, x;
|
|
|
+ if (mimio->pktbuf.q - mimio->pktbuf.p < (down ? 4 : 3))
|
|
|
+ return 1; /* partial pkt */
|
|
|
+
|
|
|
+ if (down) {
|
|
|
+ x = *mimio->pktbuf.p ^ *(mimio->pktbuf.p + 1) ^
|
|
|
+ *(mimio->pktbuf.p + 2);
|
|
|
+ if (x != *(mimio->pktbuf.p + 3)) {
|
|
|
+ dev_dbg(&mimio->idev->dev, "EV_PEN%s: bad xsum.\n",
|
|
|
+ down ? "DOWN":"UP");
|
|
|
+ /* skip this event data */
|
|
|
+ mimio->pktbuf.p += 4;
|
|
|
+ /* decode any remaining events */
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+ penid = mimio->pktbuf.instr = *(mimio->pktbuf.p + 2);
|
|
|
+ if (penid > MIMIO_PEN_MAX) {
|
|
|
+ dev_dbg(&mimio->idev->dev,
|
|
|
+ "Unmapped penID (not in [0, %d]): %d\n",
|
|
|
+ MIMIO_PEN_MAX, (int)mimio->pktbuf.instr);
|
|
|
+ penid = mimio->pktbuf.instr = 0;
|
|
|
+ }
|
|
|
+ mimio->last_pen_down = penid;
|
|
|
+ } else {
|
|
|
+ penid = mimio->last_pen_down;
|
|
|
+ }
|
|
|
+ dev_dbg(&mimio->idev->dev, "%s (id %d, code %d) %s.\n", instr[penid],
|
|
|
+ instr_ofst[penid], penid, down ? "down" : "up");
|
|
|
+
|
|
|
+ if (instr_ofst[penid] >= 0) {
|
|
|
+ int code = BTN_TOOL_PEN + instr_ofst[penid];
|
|
|
+ int value = down ? DOWNVALUE : UPVALUE;
|
|
|
+ if (code > KEY_MAX)
|
|
|
+ dev_dbg(&mimio->idev->dev, "input_event will ignore "
|
|
|
+ "-- code (%d) > KEY_MAX\n", code);
|
|
|
+ if (!test_bit(code, mimio->idev->keybit))
|
|
|
+ dev_dbg(&mimio->idev->dev, "input_event will ignore "
|
|
|
+ "-- bit for code (%d) not enabled\n", code);
|
|
|
+ if (!!test_bit(code, mimio->idev->key) == value)
|
|
|
+ dev_dbg(&mimio->idev->dev, "input_event will ignore "
|
|
|
+ "-- bit for code (%d) already set to %d\n",
|
|
|
+ code, value);
|
|
|
+ if (value != DOWNVALUE) {
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_report_key(mimio->idev, code, value);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ } else {
|
|
|
+ /* wait until we get some coordinates */
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ dev_dbg(&mimio->idev->dev, "penID offset[%d] == %d is < 0 "
|
|
|
+ "- not sending\n", penid, instr_ofst[penid]);
|
|
|
+ }
|
|
|
+ mimio->pktbuf.p += down ? 4 : 3; /* 3 for up, 4 for down */
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * Stay tuned for partial-packet excitement.
|
|
|
+ *
|
|
|
+ * This routine buffers data packets received from the mimio device
|
|
|
+ * in the mimio's data space. This buffering is necessary because
|
|
|
+ * the mimio's in endpoint can serve us partial packets of data, and
|
|
|
+ * we want the driver to support the servicing of multiple mimios.
|
|
|
+ * Empirical evidence gathered so far suggests that the method of
|
|
|
+ * buffering packet data in the mimio's data space works. Previous
|
|
|
+ * versions of this driver did not buffer packet data in each mimio's
|
|
|
+ * data-space, and were therefore not able to service multiple mimios.
|
|
|
+ * Note that since the caller of this routine is running in interrupt
|
|
|
+ * context, care needs to be taken to ensure that this routine does not
|
|
|
+ * become bloated, and it may be that another spinlock is needed in each
|
|
|
+ * mimio to guard the buffered packet data properly.
|
|
|
+ */
|
|
|
+static void mimio_rx_handler(struct mimio *mimio,
|
|
|
+ unsigned char *data,
|
|
|
+ unsigned int nbytes)
|
|
|
+{
|
|
|
+ struct device *dev = &mimio->idev->dev;
|
|
|
+ unsigned int x;
|
|
|
+ unsigned int y;
|
|
|
+ static const char * const instr[] = {
|
|
|
+ "?0",
|
|
|
+ "black pen", "blue pen", "green pen", "red pen",
|
|
|
+ "brown pen", "orange pen", "purple pen", "yellow pen",
|
|
|
+ "big eraser", "lil eraser",
|
|
|
+ "?11", "?12", "?13", "?14", "?15", "?16",
|
|
|
+ "mimio interactive", "interactive button1",
|
|
|
+ "interactive button2"
|
|
|
+ };
|
|
|
+
|
|
|
+ /* Mimio Interactive gives:
|
|
|
+ * down: [0x22 0x01 0x11 0x32 0x24]
|
|
|
+ * b1 : [0x22 0x01 0x12 0x31 0x24]
|
|
|
+ * b2 : [0x22 0x01 0x13 0x30 0x24]
|
|
|
+ */
|
|
|
+ static const int instr_ofst[] = {
|
|
|
+ -1,
|
|
|
+ 0, 1, 2, 3,
|
|
|
+ 9, 9, 9, 9,
|
|
|
+ 4, 5,
|
|
|
+ -1, -1, -1, -1, -1, -1,
|
|
|
+ 6, 7, 8,
|
|
|
+ };
|
|
|
+
|
|
|
+ memcpy(mimio->pktbuf.q, data, nbytes);
|
|
|
+ mimio->pktbuf.q += nbytes;
|
|
|
+
|
|
|
+ while (mimio->pktbuf.p < mimio->pktbuf.q) {
|
|
|
+ int t = *mimio->pktbuf.p;
|
|
|
+ switch (t) {
|
|
|
+ case MIMIO_EV_PENUP:
|
|
|
+ case MIMIO_EV_PENDOWN:
|
|
|
+ if (handle_mimio_rx_penupdown(mimio,
|
|
|
+ t == MIMIO_EV_PENDOWN,
|
|
|
+ instr, instr_ofst))
|
|
|
+ return; /* partial packet */
|
|
|
+ break;
|
|
|
+
|
|
|
+ case MIMIO_EV_PENDATA:
|
|
|
+ if (mimio->pktbuf.q - mimio->pktbuf.p < 6)
|
|
|
+ /* partial pkt */
|
|
|
+ return;
|
|
|
+ x = *mimio->pktbuf.p ^ *(mimio->pktbuf.p + 1) ^
|
|
|
+ *(mimio->pktbuf.p + 2) ^
|
|
|
+ *(mimio->pktbuf.p + 3) ^
|
|
|
+ *(mimio->pktbuf.p + 4);
|
|
|
+ if (x != *(mimio->pktbuf.p + 5)) {
|
|
|
+ dev_dbg(dev, "EV_PENDATA: bad xsum.\n");
|
|
|
+ mimio->pktbuf.p += 6; /* skip this event data */
|
|
|
+ break; /* decode any remaining events */
|
|
|
+ }
|
|
|
+ x = *(mimio->pktbuf.p + 1);
|
|
|
+ x <<= 8;
|
|
|
+ x |= *(mimio->pktbuf.p + 2);
|
|
|
+ y = *(mimio->pktbuf.p + 3);
|
|
|
+ y <<= 8;
|
|
|
+ y |= *(mimio->pktbuf.p + 4);
|
|
|
+ dev_dbg(dev, "coord: (%d, %d)\n", x, y);
|
|
|
+ if (instr_ofst[mimio->pktbuf.instr] >= 0) {
|
|
|
+ int code = BTN_TOOL_PEN +
|
|
|
+ instr_ofst[mimio->last_pen_down];
|
|
|
+#if 0
|
|
|
+ /* Utter hack to ensure we get forwarded _AND_
|
|
|
+ * so we can identify when a complete signal is
|
|
|
+ * received */
|
|
|
+ mimio->idev->abs[ABS_Y] = -1;
|
|
|
+ mimio->idev->abs[ABS_X] = -1;
|
|
|
+#endif
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_report_abs(mimio->idev, ABS_X, x);
|
|
|
+ input_report_abs(mimio->idev, ABS_Y, y);
|
|
|
+ /* fake a penup */
|
|
|
+ change_bit(code, mimio->idev->key);
|
|
|
+ input_report_key(mimio->idev,
|
|
|
+ code,
|
|
|
+ DOWNVALUE);
|
|
|
+ /* always sync here */
|
|
|
+ mimio->idev->sync = 0;
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ }
|
|
|
+ mimio->pktbuf.p += 6;
|
|
|
+ break;
|
|
|
+ case MIMIO_EV_MEMRESET:
|
|
|
+ if (mimio->pktbuf.q - mimio->pktbuf.p < 7)
|
|
|
+ /* partial pkt */
|
|
|
+ return;
|
|
|
+ dev_dbg(dev, "mem-reset.\n");
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_0, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_0, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ mimio->pktbuf.p += 7;
|
|
|
+ break;
|
|
|
+ case MIMIO_EV_ACC:
|
|
|
+ if (mimio->pktbuf.q - mimio->pktbuf.p < 4)
|
|
|
+ /* partial pkt */
|
|
|
+ return;
|
|
|
+ x = *mimio->pktbuf.p ^ *(mimio->pktbuf.p + 1) ^
|
|
|
+ *(mimio->pktbuf.p + 2);
|
|
|
+ if (x != *(mimio->pktbuf.p + 3)) {
|
|
|
+ dev_dbg(dev, "EV_ACC: bad xsum.\n");
|
|
|
+ mimio->pktbuf.p += 4; /* skip this event data */
|
|
|
+ break; /* decode any remaining events */
|
|
|
+ }
|
|
|
+ switch (*(mimio->pktbuf.p + 2)) {
|
|
|
+ case ACC_NEWPAGE:
|
|
|
+ dev_dbg(&mimio->idev->dev, "new-page.\n");
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_1, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_1, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ break;
|
|
|
+ case ACC_TAGPAGE:
|
|
|
+ dev_dbg(&mimio->idev->dev, "tag-page.\n");
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_2, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_2, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ break;
|
|
|
+ case ACC_PRINTPAGE:
|
|
|
+ dev_dbg(&mimio->idev->dev, "print-page.\n");
|
|
|
+ /* input_regs(mimio->idev, regs);*/
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_3, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_3, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ break;
|
|
|
+ case ACC_MAXIMIZE:
|
|
|
+ dev_dbg(&mimio->idev->dev,
|
|
|
+ "maximize-window.\n");
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_4, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_4, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ break;
|
|
|
+ case ACC_FINDCTLPNL:
|
|
|
+ dev_dbg(&mimio->idev->dev, "find-ctl-panel.\n");
|
|
|
+ /* input_regs(mimio->idev, regs); */
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_5, 1);
|
|
|
+ input_event(mimio->idev, EV_KEY, BTN_5, 0);
|
|
|
+ input_sync(mimio->idev);
|
|
|
+ break;
|
|
|
+ case ACC_DONE:
|
|
|
+ dev_dbg(&mimio->idev->dev, "acc-done.\n");
|
|
|
+ /* no event is dispatched to the input
|
|
|
+ * subsystem for this device event.
|
|
|
+ */
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ dev_dbg(dev, "unknown acc event.\n");
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ mimio->pktbuf.p += 4;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ mimio->pktbuf.p++;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * No partial event was received, so reset mimio's pktbuf ptrs.
|
|
|
+ */
|
|
|
+ mimio->pktbuf.p = mimio->pktbuf.q = mimio->pktbuf.buf;
|
|
|
+}
|
|
|
+
|
|
|
+static int mimio_tx(struct mimio *mimio, const char *buf, int nbytes)
|
|
|
+{
|
|
|
+ int rslt;
|
|
|
+ int timeout;
|
|
|
+ unsigned long flags;
|
|
|
+ DECLARE_WAITQUEUE(wait, current);
|
|
|
+
|
|
|
+ if (!(isvalidtxsize(nbytes))) {
|
|
|
+ dev_err(&mimio->idev->dev, "invalid arg: nbytes: %d.\n",
|
|
|
+ nbytes);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Init the out urb and copy the data to send.
|
|
|
+ */
|
|
|
+ mimio->out.urb->dev = mimio->udev;
|
|
|
+ mimio->out.urb->transfer_buffer_length = nbytes;
|
|
|
+ memcpy(mimio->out.urb->transfer_buffer, buf, nbytes);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Send the data.
|
|
|
+ */
|
|
|
+ spin_lock_irqsave(&mimio->txlock, flags);
|
|
|
+ mimio->txflags = MIMIO_TXWAIT;
|
|
|
+ rslt = usb_submit_urb(mimio->out.urb, GFP_ATOMIC);
|
|
|
+ spin_unlock_irqrestore(&mimio->txlock, flags);
|
|
|
+ dev_dbg(&mimio->idev->dev, "rslt: %d.\n", rslt);
|
|
|
+
|
|
|
+ if (rslt) {
|
|
|
+ dev_err(&mimio->idev->dev, "usb_submit_urb failure: %d.\n",
|
|
|
+ rslt);
|
|
|
+ return rslt;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Wait for completion to be signalled (the mimio_irq_out
|
|
|
+ * completion routine will or MIMIO_TXDONE in with txflags).
|
|
|
+ */
|
|
|
+ timeout = HZ;
|
|
|
+ set_current_state(TASK_INTERRUPTIBLE);
|
|
|
+ add_wait_queue(&mimio->waitq, &wait);
|
|
|
+
|
|
|
+ while (timeout && ((mimio->txflags & MIMIO_TXDONE) == 0)) {
|
|
|
+ timeout = schedule_timeout(timeout);
|
|
|
+ rmb();
|
|
|
+ }
|
|
|
+
|
|
|
+ if ((mimio->txflags & MIMIO_TXDONE) == 0)
|
|
|
+ dev_dbg(&mimio->idev->dev, "tx timed out.\n");
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Now that completion has been signalled,
|
|
|
+ * unlink the urb so that it can be recycled.
|
|
|
+ */
|
|
|
+ set_current_state(TASK_RUNNING);
|
|
|
+ remove_wait_queue(&mimio->waitq, &wait);
|
|
|
+ usb_unlink_urb(mimio->out.urb);
|
|
|
+
|
|
|
+ return rslt;
|
|
|
+}
|
|
|
+
|
|
|
+static int __init mimio_init(void)
|
|
|
+{
|
|
|
+ int rslt;
|
|
|
+
|
|
|
+ rslt = usb_register(&mimio_driver);
|
|
|
+ if (rslt != 0) {
|
|
|
+ err("%s: usb_register failure: %d", __func__, rslt);
|
|
|
+ return rslt;
|
|
|
+ }
|
|
|
+
|
|
|
+ printk(KERN_INFO KBUILD_MODNAME ":"
|
|
|
+ DRIVER_DESC " " DRIVER_VERSION "\n");
|
|
|
+ return rslt;
|
|
|
+}
|
|
|
+
|
|
|
+static void __exit mimio_exit(void)
|
|
|
+{
|
|
|
+ usb_deregister(&mimio_driver);
|
|
|
+}
|
|
|
+
|
|
|
+module_init(mimio_init);
|
|
|
+module_exit(mimio_exit);
|
|
|
+
|
|
|
+MODULE_AUTHOR(DRIVER_AUTHOR);
|
|
|
+MODULE_DESCRIPTION(DRIVER_DESC);
|
|
|
+MODULE_LICENSE("GPL");
|