Am Dienstag, 6. Februar 2007 16:15 schrieb Jiri Kosina:
> On Tue, 6 Feb 2007, Oliver Neukum wrote:
> 
> > this fixes the issue with output in the hid driver. I can now get 
> > keyboards to sleep. It is still not perfect, but an important step. 
> > Comments?
> 
> Hi Oliver,
> 
> I am going to look at it more thoroughly soon, just a quick note - when I 
> plug in the USB keyboard, I get

Yes, sorry, noticed too late.
A fixed patch is included here.

        Regards
                Oliver
----

--- linux-2.6.20/include/linux/hid.h    2007-02-06 14:14:56.000000000 +0100
+++ linux-2.6.20-autosusp/include/linux/hid.h   2007-02-06 10:31:19.000000000 
+0100
@@ -389,6 +389,8 @@
 #define HID_RESET_PENDING      4
 #define HID_SUSPENDED          5
 #define HID_CLEAR_HALT         6
+#define HID_IDLE               7
+#define HID_REPORTED_IDLE      8
 
 struct hid_input {
        struct list_head list;
--- linux-2.6.20/drivers/usb/input/usbhid.h     2007-02-06 14:14:49.000000000 
+0100
+++ linux-2.6.20-autosusp/drivers/usb/input/usbhid.h    2007-02-06 
10:31:19.000000000 +0100
@@ -72,12 +72,15 @@
        dma_addr_t outbuf_dma;                                          /* 
Output buffer dma */
        spinlock_t outlock;                                             /* 
Output fifo spinlock */
 
-       unsigned long iofl;                                             /* I/O 
flags (CTRL_RUNNING, OUT_RUNNING) */
-       struct timer_list io_retry;                                     /* 
Retry timer */
-       unsigned long stop_retry;                                       /* Time 
to give up, in jiffies */
-       unsigned int retry_delay;                                       /* 
Delay length in ms */
-       struct work_struct reset_work;                                  /* Task 
context for resets */
-
+       unsigned long iofl;                                     /* I/O flags 
(CTRL_RUNNING, OUT_RUNNING) */
+       struct timer_list io_retry;                             /* Retry timer 
*/
+       struct timer_list idle_timer;                           /* timer to 
determine idleness */
+       unsigned long stop_retry;                               /* Time to give 
up, in jiffies */
+       unsigned int idle_time;                                 /* Time to 
determine idleness, in seconds */
+       unsigned int retry_delay;                               /* Delay length 
in ms */
+       struct work_struct reset_work;                          /* Task context 
for resets */
+       struct work_struct idle_work;                           /* Task context 
for reporting idleness */
+       struct work_struct restart_work;                        /* waking up 
for output to be done in task context */
 };
 
 #define        hid_to_usb_dev(hid_dev) \
--- linux-2.6.20/drivers/usb/input/hid-core.c   2007-02-06 14:14:48.000000000 
+0100
+++ linux-2.6.20-autosusp/drivers/usb/input/hid-core.c  2007-02-06 
15:32:53.000000000 +0100
@@ -5,6 +5,7 @@
  *  Copyright (c) 2000-2005 Vojtech Pavlik <[EMAIL PROTECTED]>
  *  Copyright (c) 2005 Michael Haboustak <[EMAIL PROTECTED]> for Concept2, Inc
  *  Copyright (c) 2006 Jiri Kosina
+ *  Copyright (c) 2007 Oliver Neukum
  */
 
 /*
@@ -56,11 +57,45 @@
 module_param_named(mousepoll, hid_mousepoll_interval, uint, 0644);
 MODULE_PARM_DESC(mousepoll, "Polling interval of mice");
 
+static spinlock_t idle_guard = SPIN_LOCK_UNLOCKED;
+static DEFINE_MUTEX(hid_open_mut);
+
+/*
+ * sysfs idle time handling
+ */
+
+static ssize_t show_idle_time(struct device *dev, struct device_attribute 
*attr, char *buf)
+{
+       struct usb_interface *intf = to_usb_interface(dev);
+       struct hid_device *hh = usb_get_intfdata(intf);
+       struct usbhid_device *uh = hh->driver_data;
+
+       return sprintf(buf, "%d\n", uh->idle_time);
+}
+
+static ssize_t set_idle_time(struct device *dev, struct device_attribute 
*attr, const char *buf, size_t count)
+{
+       struct usb_interface *intf = to_usb_interface(dev);
+       struct hid_device *hh = usb_get_intfdata(intf);
+       struct usbhid_device *uh = hh->driver_data;
+       int temp = simple_strtoul(buf, NULL, 10);
+
+       spin_lock_irq(&idle_guard);
+       uh->idle_time = temp;
+       spin_unlock_irq(&idle_guard);
+
+       return count;
+}
+
+static DEVICE_ATTR(idle_time, S_IWUGO | S_IRUGO, show_idle_time, 
set_idle_time);
+
 /*
  * Input submission and I/O error handler.
  */
 
 static void hid_io_error(struct hid_device *hid);
+static int hid_submit_out(struct hid_device *hid);
+static int hid_submit_ctrl(struct hid_device *hid);
 
 /* Start up the input URB */
 static int hid_start_in(struct hid_device *hid)
@@ -169,6 +204,44 @@
        spin_unlock_irqrestore(&usbhid->inlock, flags);
 }
 
+static int usbhid_restart_out_queue(struct usbhid_device *usbhid)
+{
+       struct hid_device *hid = usb_get_intfdata(usbhid->intf);
+       int kicked;
+
+       WARN_ON(hid == NULL);
+       if (!hid)
+               return 0;
+
+       if ((kicked = (usbhid->outhead != usbhid->outtail))) {
+               dbg("Kicking head %d tail %d", usbhid->outhead, 
usbhid->outtail);
+               if (hid_submit_out(hid)) {
+                       clear_bit(HID_OUT_RUNNING, &usbhid->iofl);
+                       wake_up(&hid->wait);
+               }
+       }
+       return kicked;
+}
+
+static int usbhid_restart_ctrl_queue(struct usbhid_device *usbhid)
+{
+       struct hid_device *hid = usb_get_intfdata(usbhid->intf);
+       int kicked;
+
+       WARN_ON(hid == NULL);
+       if (!hid)
+               return 0;
+
+       if ((kicked = (usbhid->ctrlhead != usbhid->ctrltail))) {
+               dbg("Kicking head %d tail %d", usbhid->ctrlhead, 
usbhid->ctrltail);
+               if (hid_submit_ctrl(hid)) {
+                       clear_bit(HID_CTRL_RUNNING, &usbhid->iofl);
+                       wake_up(&hid->wait);
+               }
+       }
+       return kicked;
+}
+
 /*
  * Input interrupt completion handler.
  */
@@ -179,6 +252,8 @@
        struct usbhid_device    *usbhid = hid->driver_data;
        int                     status;
 
+       /*data recieved, errors ignored as they are handled, device not idle */
+       clear_bit(HID_IDLE, &usbhid->iofl);
        switch (urb->status) {
                case 0:                 /* success */
                        usbhid->retry_delay = 0;
@@ -242,9 +317,11 @@
        struct hid_report *report;
        struct usbhid_device *usbhid = hid->driver_data;
 
+       WARN_ON(usbhid == NULL);
        report = usbhid->out[usbhid->outtail];
-
+       WARN_ON(report == NULL);
        hid_output_report(report, usbhid->outbuf);
+       BUG_ON(usbhid->urbout == NULL);
        usbhid->urbout->transfer_buffer_length = ((report->size - 1) >> 3) + 1 
+ (report->id > 0);
        usbhid->urbout->dev = hid_to_usb_dev(hid);
 
@@ -339,11 +416,7 @@
        else
                usbhid->outtail = (usbhid->outtail + 1) & (HID_OUTPUT_FIFO_SIZE 
- 1);
 
-       if (usbhid->outhead != usbhid->outtail) {
-               if (hid_submit_out(hid)) {
-                       clear_bit(HID_OUT_RUNNING, &usbhid->iofl);
-                       wake_up(&hid->wait);
-               }
+       if (usbhid_restart_out_queue(usbhid)) {
                spin_unlock_irqrestore(&usbhid->outlock, flags);
                return;
        }
@@ -389,11 +462,7 @@
        else
                usbhid->ctrltail = (usbhid->ctrltail + 1) & 
(HID_CONTROL_FIFO_SIZE - 1);
 
-       if (usbhid->ctrlhead != usbhid->ctrltail) {
-               if (hid_submit_ctrl(hid)) {
-                       clear_bit(HID_CTRL_RUNNING, &usbhid->iofl);
-                       wake_up(&hid->wait);
-               }
+       if (usbhid_restart_ctrl_queue(usbhid)) {
                spin_unlock_irqrestore(&usbhid->ctrllock, flags);
                return;
        }
@@ -403,9 +472,45 @@
        wake_up(&hid->wait);
 }
 
-void usbhid_submit_report(struct hid_device *hid, struct hid_report *report, 
unsigned char dir)
+static int usbhid_queue_report_out(struct hid_device *hid, struct hid_report 
*report)
+{
+       struct usbhid_device *usbhid = hid->driver_data;
+       int head;
+
+       if ((head = (usbhid->outhead + 1) & (HID_OUTPUT_FIFO_SIZE - 1)) == 
usbhid->outtail) {
+               warn("output queue full");
+               return 0;
+       }
+
+       usbhid->out[usbhid->outhead] = report;
+       usbhid->outhead = head;
+
+       return usbhid->outhead < usbhid->outtail ?
+               usbhid->outtail - usbhid->outhead :
+               HID_OUTPUT_FIFO_SIZE - usbhid->outhead + usbhid->outtail;
+}
+
+static int usbhid_queue_report_ctrl(struct hid_device *hid, struct hid_report 
*report, unsigned char dir)
 {
+       struct usbhid_device *usbhid = hid->driver_data;
        int head;
+
+       if ((head = (usbhid->ctrlhead + 1) & (HID_CONTROL_FIFO_SIZE - 1)) == 
usbhid->ctrltail) {
+               warn("control queue full");
+               return 0;
+       }
+
+       usbhid->ctrl[usbhid->ctrlhead].report = report;
+       usbhid->ctrl[usbhid->ctrlhead].dir = dir;
+       usbhid->ctrlhead = head;
+
+       return usbhid->ctrlhead < usbhid->ctrltail ?
+               usbhid->ctrltail - usbhid->ctrlhead :
+               HID_CONTROL_FIFO_SIZE - usbhid->ctrlhead + usbhid->ctrltail;
+}
+
+void usbhid_submit_report(struct hid_device *hid, struct hid_report *report, 
unsigned char dir)
+{
        unsigned long flags;
        struct usbhid_device *usbhid = hid->driver_data;
 
@@ -416,47 +521,46 @@
 
                spin_lock_irqsave(&usbhid->outlock, flags);
 
-               if ((head = (usbhid->outhead + 1) & (HID_OUTPUT_FIFO_SIZE - 1)) 
== usbhid->outtail) {
-                       spin_unlock_irqrestore(&usbhid->outlock, flags);
-                       warn("output queue full");
-                       return;
-               }
-
-               usbhid->out[usbhid->outhead] = report;
-               usbhid->outhead = head;
+               usbhid_queue_report_out(hid, report);
 
                if (!test_and_set_bit(HID_OUT_RUNNING, &usbhid->iofl))
                        if (hid_submit_out(hid))
                                clear_bit(HID_OUT_RUNNING, &usbhid->iofl);
 
                spin_unlock_irqrestore(&usbhid->outlock, flags);
-               return;
-       }
+       } else {
+               spin_lock_irqsave(&usbhid->ctrllock, flags);
 
-       spin_lock_irqsave(&usbhid->ctrllock, flags);
+               usbhid_queue_report_ctrl(hid, report, dir);
+
+               if (!test_and_set_bit(HID_CTRL_RUNNING, &usbhid->iofl))
+                       if (hid_submit_ctrl(hid))
+                               clear_bit(HID_CTRL_RUNNING, &usbhid->iofl);
 
-       if ((head = (usbhid->ctrlhead + 1) & (HID_CONTROL_FIFO_SIZE - 1)) == 
usbhid->ctrltail) {
                spin_unlock_irqrestore(&usbhid->ctrllock, flags);
-               warn("control queue full");
-               return;
        }
+}
 
-       usbhid->ctrl[usbhid->ctrlhead].report = report;
-       usbhid->ctrl[usbhid->ctrlhead].dir = dir;
-       usbhid->ctrlhead = head;
-
-       if (!test_and_set_bit(HID_CTRL_RUNNING, &usbhid->iofl))
-               if (hid_submit_ctrl(hid))
-                       clear_bit(HID_CTRL_RUNNING, &usbhid->iofl);
+static int usbhid_queue_report(struct hid_device *hid, struct hid_report 
*report, unsigned char dir)
+{
+       struct usbhid_device *usbhid = hid->driver_data;
+       int rv;
 
-       spin_unlock_irqrestore(&usbhid->ctrllock, flags);
+       if (usbhid->urbout && dir == USB_DIR_OUT && report->type == 
HID_OUTPUT_REPORT) {
+               rv = usbhid_queue_report_out(hid, report);
+       } else {
+               rv = usbhid_queue_report_ctrl(hid, report, dir);
+       }
+       return rv;
 }
 
 static int usb_hidinput_input_event(struct input_dev *dev, unsigned int type, 
unsigned int code, int value)
 {
        struct hid_device *hid = dev->private;
+       struct usbhid_device *usbhid = hid->driver_data;
        struct hid_field *field;
-       int offset;
+       unsigned long flags;
+       int offset, used;
 
        if (type == EV_FF)
                return input_ff_event(dev, type, code, value);
@@ -469,8 +573,24 @@
                return -1;
        }
 
+       spin_lock_irqsave(&usbhid->outlock, flags);
+       spin_lock(&usbhid->ctrllock);
+       clear_bit(HID_IDLE, &usbhid->iofl);
        hid_set_field(field, offset, value);
-       usbhid_submit_report(hid, field->report, USB_DIR_OUT);
+       if (!test_bit(HID_REPORTED_IDLE, &usbhid->iofl)) {
+               /* the device isn't idle, we do the output now*/
+               spin_unlock(&usbhid->ctrllock);
+               spin_unlock_irqrestore(&usbhid->outlock, flags);
+               usbhid_submit_report(hid, field->report, USB_DIR_OUT);
+       } else {
+               /* the device is idle, queue the request
+               and wake it up if the queue fills up too much*/
+               used = usbhid_queue_report(hid, field->report, USB_DIR_OUT);
+               if ( 3 < HID_OUTPUT_FIFO_SIZE - used)
+                       schedule_work(&usbhid->restart_work);
+               spin_unlock(&usbhid->ctrllock);
+               spin_unlock_irqrestore(&usbhid->outlock, flags);
+       }
 
        return 0;
 }
@@ -514,7 +634,20 @@
 
 int usbhid_open(struct hid_device *hid)
 {
-       ++hid->open;
+       struct usbhid_device *usbhid = hid->driver_data;
+       int res;
+
+       clear_bit(HID_IDLE, &usbhid->iofl);
+       mutex_lock(&hid_open_mut);
+       if (!hid->open++) {
+               res = usb_autopm_get_interface(usbhid->intf);
+               if (res < 0) {
+                       hid->open--;
+                       mutex_unlock(&hid_open_mut);
+                       return -EIO;
+               }
+       }
+       mutex_unlock(&hid_open_mut);
        if (hid_start_in(hid))
                hid_io_error(hid);
        return 0;
@@ -524,8 +657,13 @@
 {
        struct usbhid_device *usbhid = hid->driver_data;
 
-       if (!--hid->open)
+       if (!--hid->open) {
                usb_kill_urb(usbhid->urbin);
+               del_timer_sync(&usbhid->idle_timer);
+               flush_scheduled_work();
+               if (!test_and_clear_bit(HID_REPORTED_IDLE, &usbhid->iofl))
+                       usb_autopm_put_interface(usbhid->intf);
+       }
 }
 
 static int hidinput_open(struct input_dev *dev)
@@ -1005,6 +1143,80 @@
        return 0;
 }
 
+void hid_add_idle_timer(struct usbhid_device *uhid)
+{
+       uhid->idle_timer.expires = jiffies + uhid->idle_time * HZ;
+       add_timer(&uhid->idle_timer);
+printk(KERN_ERR"Added idle timer for firing in %d seconds\n", uhid->idle_time);
+}
+
+static void hid_mod_idle_timer(struct usbhid_device *uhid)
+{
+       if (usb_get_intfdata(uhid->intf) && uhid->idle_time) {
+               mod_timer(&uhid->idle_timer, jiffies + uhid->idle_time * HZ);
+               dbg("Modded idle timer for firing in %d seconds", 
uhid->idle_time);
+       }
+}
+
+static void usbhid_restart_queues(struct usbhid_device *usbhid)
+{
+       if (usbhid->urbout)
+               usbhid_restart_out_queue(usbhid);
+       usbhid_restart_ctrl_queue(usbhid);
+}
+
+static void __hid_report_idle(struct work_struct *work)
+{
+       struct usbhid_device *usbhid =
+               container_of(work, struct usbhid_device, idle_work);
+
+       usb_autopm_put_interface(usbhid->intf);
+}
+
+static void __usbhid_restart_queues(struct work_struct *work)
+{
+       struct usbhid_device *usbhid =
+               container_of(work, struct usbhid_device, restart_work);
+
+       usb_autopm_get_interface(usbhid->intf);
+       usbhid_restart_queues(usbhid);
+       usb_autopm_put_interface(usbhid->intf);
+}
+
+static void hid_report_idle(struct usbhid_device *hid)
+{
+       struct usb_interface *intf = hid->intf;
+
+       if (!test_bit(HID_SUSPENDED, &hid->iofl)
+       && !test_bit(HID_CTRL_RUNNING, &hid->iofl)
+       && !test_bit(HID_OUT_RUNNING, &hid->iofl)) {
+               set_bit(HID_REPORTED_IDLE, &hid->iofl);
+               if (usb_get_intfdata(intf))
+                       schedule_work(&hid->idle_work);
+       }
+
+}
+
+static void hid_check_idle(unsigned long arg)
+{
+       struct usbhid_device *hid = (struct usbhid_device *)arg;
+       unsigned long flags;
+       int state;
+
+       spin_lock_irqsave(&hid->inlock, flags);
+       spin_lock(&hid->outlock);
+       state = test_and_set_bit(HID_IDLE, &hid->iofl);
+
+       if (!state) {
+               hid_mod_idle_timer(hid);
+       } else {
+               hid_report_idle(hid);
+               dbg("Detected idleness");
+       }
+       spin_unlock(&hid->outlock);
+       spin_unlock_irqrestore(&hid->inlock, flags);
+}
+
 static void hid_free_buffers(struct usb_device *dev, struct hid_device *hid)
 {
        struct usbhid_device *usbhid = hid->driver_data;
@@ -1183,17 +1395,23 @@
        init_waitqueue_head(&hid->wait);
 
        INIT_WORK(&usbhid->reset_work, hid_reset);
+       INIT_WORK(&usbhid->idle_work, __hid_report_idle);
+       INIT_WORK(&usbhid->restart_work, __usbhid_restart_queues);
        setup_timer(&usbhid->io_retry, hid_retry_timeout, (unsigned long) hid);
+       setup_timer(&usbhid->idle_timer, hid_check_idle, (unsigned long) 
usbhid);
 
        spin_lock_init(&usbhid->inlock);
        spin_lock_init(&usbhid->outlock);
        spin_lock_init(&usbhid->ctrllock);
 
+       set_bit(HID_IDLE, &usbhid->iofl);
+
        hid->version = le16_to_cpu(hdesc->bcdHID);
        hid->country = hdesc->bCountryCode;
        hid->dev = &intf->dev;
        usbhid->intf = intf;
        usbhid->ifnum = interface->desc.bInterfaceNumber;
+       usbhid->idle_time = 10; //FIXME
 
        hid->name[0] = 0;
 
@@ -1263,6 +1481,8 @@
 
        usbhid = hid->driver_data;
 
+       device_remove_file(&intf->dev, &dev_attr_idle_time);
+
        spin_lock_irq(&usbhid->inlock); /* Sync with error handler */
        usb_set_intfdata(intf, NULL);
        spin_unlock_irq(&usbhid->inlock);
@@ -1271,6 +1491,7 @@
        usb_kill_urb(usbhid->urbctrl);
 
        del_timer_sync(&usbhid->io_retry);
+       del_timer_sync(&usbhid->idle_timer);
        flush_scheduled_work();
 
        if (hid->claimed & HID_CLAIMED_INPUT)
@@ -1299,6 +1520,9 @@
        if (!(hid = usb_hid_configure(intf)))
                return -ENODEV;
 
+       usb_set_intfdata(intf, hid);
+       intf->needs_remote_wakeup = 1;
+
        usbhid_init_reports(hid);
        hid_dump_device(hid);
 
@@ -1307,7 +1531,6 @@
        if (!hiddev_connect(hid))
                hid->claimed |= HID_CLAIMED_HIDDEV;
 
-       usb_set_intfdata(intf, hid);
 
        if (!hid->claimed) {
                printk ("HID device not claimed by input or hiddev\n");
@@ -1331,6 +1554,9 @@
        if (hid->claimed & HID_CLAIMED_HIDDEV)
                printk("hiddev%d", hid->minor);
 
+       if (hid->claimed & HID_CLAIMED_INPUT)
+               hid_add_idle_timer(hid->driver_data);
+
        c = "Device";
        for (i = 0; i < hid->maxcollection; i++) {
                if (hid->collection[i].type == HID_COLLECTION_APPLICATION &&
@@ -1342,6 +1568,7 @@
        }
 
        usb_make_path(interface_to_usbdev(intf), path, 63);
+       device_create_file(&intf->dev, &dev_attr_idle_time);
 
        printk(": USB HID v%x.%02x %s [%s] on %s\n",
                hid->version >> 8, hid->version & 0xff, c, hid->name, path);
@@ -1359,6 +1586,7 @@
        spin_unlock_irq(&usbhid->inlock);
        del_timer(&usbhid->io_retry);
        usb_kill_urb(usbhid->urbin);
+       flush_scheduled_work();
        dev_dbg(&intf->dev, "suspend\n");
        return 0;
 }
@@ -1370,8 +1598,13 @@
        int status;
 
        clear_bit(HID_SUSPENDED, &usbhid->iofl);
+       clear_bit(HID_IDLE, &usbhid->iofl);
        usbhid->retry_delay = 0;
        status = hid_start_in(hid);
+       if (test_and_clear_bit(HID_REPORTED_IDLE, &usbhid->iofl))
+               intf->pm_usage_cnt++; /* get'd deadlock */
+       usbhid_restart_queues(usbhid);
+       hid_add_idle_timer(usbhid);
        dev_dbg(&intf->dev, "resume status %d\n", status);
        return status;
 }
@@ -1410,6 +1643,7 @@
        .pre_reset =    hid_pre_reset,
        .post_reset =   hid_post_reset,
        .id_table =     hid_usb_ids,
+       .supports_autosuspend = 1,
 };
 
 static int __init hid_init(void)
--- linux-2.6.20/drivers/usb/input/hiddev.c     2007-02-06 14:14:49.000000000 
+0100
+++ linux-2.6.20-autosusp/drivers/usb/input/hiddev.c    2007-02-06 
10:31:19.000000000 +0100
@@ -496,8 +496,11 @@
                if ((report = hiddev_lookup_report(hid, &rinfo)) == NULL)
                        return -EINVAL;
 
+               if (usb_autopm_get_interface(usbhid->intf) < 0)
+                       return -EIO;
                usbhid_submit_report(hid, report, USB_DIR_IN);
                usbhid_wait_io(hid);
+               usb_autopm_put_interface(usbhid->intf);
 
                return 0;
 
@@ -511,8 +514,11 @@
                if ((report = hiddev_lookup_report(hid, &rinfo)) == NULL)
                        return -EINVAL;
 
+               if (usb_autopm_get_interface(usbhid->intf) < 0)
+                       return -EIO;
                usbhid_submit_report(hid, report, USB_DIR_OUT);
                usbhid_wait_io(hid);
+               usb_autopm_put_interface(usbhid->intf);
 
                return 0;
 

-------------------------------------------------------------------------
Using Tomcat but need to do more? Need to support web services, security?
Get stuff done quickly with pre-integrated technology to make your job easier.
Download IBM WebSphere Application Server v.1.0.1 based on Apache Geronimo
http://sel.as-us.falkag.net/sel?cmd=lnk&kid=120709&bid=263057&dat=121642
_______________________________________________
linux-usb-devel@lists.sourceforge.net
To unsubscribe, use the last form field at:
https://lists.sourceforge.net/lists/listinfo/linux-usb-devel

Reply via email to