Hi, this is the newest version of autosuspend for HID devices. It fixes:
- Jiri: shifting code around - forgot to kill output urbs upon suspend - pre/post_reset methods Regards Oliver ------ --- linux-2.6.22-rc2/include/linux/hid.h 2007-05-22 14:50:58.000000000 +0200 +++ linux-2.6.22-rc2-autosuspend/include/linux/hid.h 2007-05-22 15:04:41.000000000 +0200 @@ -401,6 +401,7 @@ struct hid_control_fifo { #define HID_RESET_PENDING 4 #define HID_SUSPENDED 5 #define HID_CLEAR_HALT 6 +#define HID_REPORTED_IDLE 7 struct hid_input { struct list_head list; @@ -500,6 +501,7 @@ void hid_input_field(struct hid_device * void hid_output_report(struct hid_report *report, __u8 *data); void hid_free_device(struct hid_device *device); struct hid_device *hid_parse_report(__u8 *start, unsigned size); +int hidinput_has_ff(struct hid_device *hid); /* HID quirks API */ u32 usbhid_lookup_quirk(const u16 idVendor, const u16 idProduct); --- linux-2.6.22-rc2/drivers/hid/usbhid/usbhid.h 2007-05-22 14:50:09.000000000 +0200 +++ linux-2.6.22-rc2-autosuspend/drivers/hid/usbhid/usbhid.h 2007-05-22 11:15:44.000000000 +0200 @@ -36,7 +36,10 @@ int usbhid_wait_io(struct hid_device* hi void usbhid_close(struct hid_device *hid); int usbhid_open(struct hid_device *hid); void usbhid_init_reports(struct hid_device *hid); -void usbhid_submit_report(struct hid_device *hid, struct hid_report *report, unsigned char dir); +void usbhid_submit_report +(struct hid_device *hid, struct hid_report *report, unsigned char dir); +int usbhid_get_power(struct hid_device *hid); +void usbhid_put_power(struct hid_device *hid); /* * USB-specific HID struct, to be pointed to @@ -44,40 +47,69 @@ void usbhid_submit_report(struct hid_dev */ struct usbhid_device { - struct hid_device *hid; /* pointer to corresponding HID dev */ - - struct usb_interface *intf; /* USB interface */ - int ifnum; /* USB interface number */ - - unsigned int bufsize; /* URB buffer size */ - - struct urb *urbin; /* Input URB */ - char *inbuf; /* Input buffer */ - dma_addr_t inbuf_dma; /* Input buffer dma */ - spinlock_t inlock; /* Input fifo spinlock */ - - struct urb *urbctrl; /* Control URB */ - struct usb_ctrlrequest *cr; /* Control request struct */ - dma_addr_t cr_dma; /* Control request struct dma */ - struct hid_control_fifo ctrl[HID_CONTROL_FIFO_SIZE]; /* Control fifo */ - unsigned char ctrlhead, ctrltail; /* Control fifo head & tail */ - char *ctrlbuf; /* Control buffer */ - dma_addr_t ctrlbuf_dma; /* Control buffer dma */ - spinlock_t ctrllock; /* Control fifo spinlock */ - - struct urb *urbout; /* Output URB */ - struct hid_report *out[HID_CONTROL_FIFO_SIZE]; /* Output pipe fifo */ - unsigned char outhead, outtail; /* Output pipe fifo head & tail */ - char *outbuf; /* Output buffer */ - 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 */ + /* pointer to corresponding HID dev */ + struct hid_device *hid; + /* USB interface */ + struct usb_interface *intf; + /* USB interface number */ + int ifnum; + + /* URB buffer size */ + unsigned int bufsize; + + /* Input URB */ + struct urb *urbin; + /* Input buffer */ + char *inbuf; + /* Input buffer dma */ + dma_addr_t inbuf_dma; + /* Input fifo spinlock */ + spinlock_t inlock; + + /* Control URB */ + struct urb *urbctrl; + /* Control request struct */ + struct usb_ctrlrequest *cr; + /* Control request struct dma */ + dma_addr_t cr_dma; + /* Control fifo */ + struct hid_control_fifo ctrl[HID_CONTROL_FIFO_SIZE]; + /* Control fifo head & tail */ + unsigned char ctrlhead, ctrltail; + /* Control buffer */ + char *ctrlbuf; + /* Control buffer dma */ + dma_addr_t ctrlbuf_dma; + + /* Output URB */ + struct urb *urbout; + /* Output pipe fifo */ + struct hid_report *out[HID_CONTROL_FIFO_SIZE]; + /* Output pipe fifo head & tail */ + unsigned char outhead, outtail; + /* Output buffer */ + char *outbuf; + /* Output buffer dma */ + dma_addr_t outbuf_dma; + + /* Output fifo spinlock */ + spinlock_t outputlock; + + /* I/O flags (CTRL_RUNNING, OUT_RUNNING) */ + unsigned long iofl; + /* Retry timer */ + struct timer_list io_retry; + /* timer to determine idleness */ + struct timer_list idle_timer; + /* Time to give up, in jiffies */ + unsigned long stop_retry; + /* Delay length in ms */ + unsigned int retry_delay; + /* Task context for resets */ + struct work_struct reset_work; + /* waking up for output to be done in task context */ + struct work_struct restart_work; }; #define hid_to_usb_dev(hid_dev) \ --- linux-2.6.22-rc2/drivers/hid/hid-input.c 2007-05-22 14:50:08.000000000 +0200 +++ linux-2.6.22-rc2-autosuspend/drivers/hid/hid-input.c 2007-05-22 15:05:09.000000000 +0200 @@ -940,6 +940,17 @@ int hidinput_find_field(struct hid_devic } EXPORT_SYMBOL_GPL(hidinput_find_field); +int hidinput_has_ff(struct hid_device *hid) +{ + struct hid_input *hidinput; + + list_for_each_entry(hidinput, &hid->inputs, list) + if (test_bit(EV_FF, hidinput->input->evbit)) + return 1; + return 0; +} +EXPORT_SYMBOL_GPL(hidinput_has_ff); + static int hidinput_open(struct input_dev *dev) { struct hid_device *hid = input_get_drvdata(dev); --- linux-2.6.22-rc2/drivers/hid/usbhid/hid-core.c 2007-05-22 14:50:09.000000000 +0200 +++ linux-2.6.22-rc2-autosuspend/drivers/hid/usbhid/hid-core.c 2007-05-22 14:58:47.000000000 +0200 @@ -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-2007 Jiri Kosina + * Copyright (c) 2007 Oliver Neukum */ /* @@ -63,8 +64,11 @@ MODULE_PARM_DESC(quirks, "Add/modify USB /* * Input submission and I/O error handler. */ +static DEFINE_MUTEX(hid_open_mut); 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) @@ -74,7 +78,7 @@ static int hid_start_in(struct hid_devic struct usbhid_device *usbhid = hid->driver_data; spin_lock_irqsave(&usbhid->inlock, flags); - if (hid->open > 0 && !test_bit(HID_SUSPENDED, &usbhid->iofl) && + if (hid->open > 0 && !test_bit(HID_REPORTED_IDLE, &usbhid->iofl) && !test_and_set_bit(HID_IN_RUNNING, &usbhid->iofl)) { rc = usb_submit_urb(usbhid->urbin, GFP_ATOMIC); if (rc != 0) @@ -178,6 +182,51 @@ done: spin_unlock_irqrestore(&usbhid->inlock, flags); } +static void usbhid_mark_busy(struct usbhid_device *usbhid) +{ + struct usb_interface *intf = usbhid->intf; + + usb_mark_last_busy(interface_to_usbdev(intf)); +} + +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. */ @@ -190,12 +239,14 @@ static void hid_irq_in(struct urb *urb) switch (urb->status) { case 0: /* success */ + usbhid_mark_busy(usbhid); usbhid->retry_delay = 0; hid_input_report(urb->context, HID_INPUT_REPORT, urb->transfer_buffer, urb->actual_length, 1); break; case -EPIPE: /* stall */ + usbhid_mark_busy(usbhid); clear_bit(HID_IN_RUNNING, &usbhid->iofl); set_bit(HID_CLEAR_HALT, &usbhid->iofl); schedule_work(&usbhid->reset_work); @@ -209,6 +260,7 @@ static void hid_irq_in(struct urb *urb) case -EPROTO: /* protocol error or unplug */ case -ETIME: /* protocol error or unplug */ case -ETIMEDOUT: /* Should never happen, but... */ + usbhid_mark_busy(usbhid); clear_bit(HID_IN_RUNNING, &usbhid->iofl); hid_io_error(hid); return; @@ -234,9 +286,11 @@ static int hid_submit_out(struct hid_dev 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); @@ -324,24 +378,20 @@ static void hid_irq_out(struct urb *urb) warn("output irq status %d received", urb->status); } - spin_lock_irqsave(&usbhid->outlock, flags); + spin_lock_irqsave(&usbhid->outputlock, flags); if (unplug) usbhid->outtail = usbhid->outhead; 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); - } - spin_unlock_irqrestore(&usbhid->outlock, flags); + if (usbhid_restart_out_queue(usbhid)) { + spin_unlock_irqrestore(&usbhid->outputlock, flags); return; } clear_bit(HID_OUT_RUNNING, &usbhid->iofl); - spin_unlock_irqrestore(&usbhid->outlock, flags); + spin_unlock_irqrestore(&usbhid->outputlock, flags); wake_up(&hid->wait); } @@ -356,7 +406,7 @@ static void hid_ctrl(struct urb *urb) unsigned long flags; int unplug = 0; - spin_lock_irqsave(&usbhid->ctrllock, flags); + spin_lock_irqsave(&usbhid->outputlock, flags); switch (urb->status) { case 0: /* success */ @@ -381,24 +431,55 @@ static void hid_ctrl(struct urb *urb) 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); - } - spin_unlock_irqrestore(&usbhid->ctrllock, flags); + if (usbhid_restart_ctrl_queue(usbhid)) { + spin_unlock_irqrestore(&usbhid->outputlock, flags); return; } clear_bit(HID_CTRL_RUNNING, &usbhid->iofl); - spin_unlock_irqrestore(&usbhid->ctrllock, flags); + spin_unlock_irqrestore(&usbhid->outputlock, flags); 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; - unsigned long flags; + + 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; +} + +static void __usbhid_submit_report(struct hid_device *hid, struct hid_report *report, unsigned char dir) +{ struct usbhid_device *usbhid = hid->driver_data; if ((hid->quirks & HID_QUIRK_NOGET) && dir == USB_DIR_IN) @@ -406,49 +487,53 @@ void usbhid_submit_report(struct hid_dev if (usbhid->urbout && dir == USB_DIR_OUT && report->type == HID_OUTPUT_REPORT) { - 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); + 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; +void usbhid_submit_report(struct hid_device *hid, struct hid_report *report, unsigned char dir) +{ + struct usbhid_device *usbhid = hid->driver_data; + unsigned long flags; - if (!test_and_set_bit(HID_CTRL_RUNNING, &usbhid->iofl)) - if (hid_submit_ctrl(hid)) - clear_bit(HID_CTRL_RUNNING, &usbhid->iofl); + spin_lock_irqsave(&usbhid->outputlock, flags); + __usbhid_submit_report(hid, report, dir); + spin_unlock_irqrestore(&usbhid->outputlock, flags); +} + +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 = input_get_drvdata(dev); + 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); @@ -461,9 +546,21 @@ static int usb_hidinput_input_event(stru return -1; } + spin_lock_irqsave(&usbhid->outputlock, flags); + 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*/ + __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_irqrestore(&usbhid->outputlock, flags); return 0; } @@ -506,9 +603,31 @@ static int hid_get_class_descriptor(stru int usbhid_open(struct hid_device *hid) { - ++hid->open; - if (hid_start_in(hid)) - hid_io_error(hid); + struct usbhid_device *usbhid = hid->driver_data; + int res; + + mutex_lock(&hid_open_mut); + if (!hid->open++) { + res = usb_autopm_get_interface(usbhid->intf); + /* the device must be awake to reliable request remote wakeup */ + if (res < 0) { + hid->open--; + mutex_unlock(&hid_open_mut); + return -EIO; + } + usbhid->intf->needs_remote_wakeup = 1; + if (hid_start_in(hid)) + hid_io_error(hid); + + /* force feedback effects last longer than the command + * to initiate them. Such devices must be suspend only + * when closed + */ + if (!hidinput_has_ff(hid)) + usb_autopm_put_interface(usbhid->intf); + } + mutex_unlock(&hid_open_mut); + return 0; } @@ -516,8 +635,24 @@ void usbhid_close(struct hid_device *hid { struct usbhid_device *usbhid = hid->driver_data; - if (!--hid->open) + mutex_lock(&hid_open_mut); + + /* protecting hid->open to make sure we don't restart + * data acquistion due to a resumption we no longer + * care about + */ + spin_lock_irq(&usbhid->inlock); + if (!--hid->open) { + spin_unlock_irq(&usbhid->inlock); usb_kill_urb(usbhid->urbin); + flush_scheduled_work(); + usbhid->intf->needs_remote_wakeup = 0; + if (hidinput_has_ff(hid)) + usb_autopm_put_interface(usbhid->intf); + } else { + spin_unlock_irq(&usbhid->inlock); + } + mutex_unlock(&hid_open_mut); } /* @@ -527,14 +662,21 @@ void usbhid_close(struct hid_device *hid void usbhid_init_reports(struct hid_device *hid) { struct hid_report *report; - struct usbhid_device *usbhid = hid->driver_data; + unsigned long flags; int err, ret; + struct usbhid_device *usbhid = hid->driver_data; - list_for_each_entry(report, &hid->report_enum[HID_INPUT_REPORT].report_list, list) - usbhid_submit_report(hid, report, USB_DIR_IN); + list_for_each_entry(report, &hid->report_enum[HID_INPUT_REPORT].report_list, list) { + spin_lock_irqsave(&usbhid->outputlock, flags); + __usbhid_submit_report(hid, report, USB_DIR_IN); + spin_unlock_irqrestore(&usbhid->outputlock, flags); + } - list_for_each_entry(report, &hid->report_enum[HID_FEATURE_REPORT].report_list, list) - usbhid_submit_report(hid, report, USB_DIR_IN); + list_for_each_entry(report, &hid->report_enum[HID_FEATURE_REPORT].report_list, list) { + spin_lock_irqsave(&usbhid->outputlock, flags); + __usbhid_submit_report(hid, report, USB_DIR_IN); + spin_unlock_irqrestore(&usbhid->outputlock, flags); + } err = 0; ret = usbhid_wait_io(hid); @@ -622,6 +764,23 @@ static int hid_alloc_buffers(struct usb_ return 0; } +static void usbhid_restart_queues(struct usbhid_device *usbhid) +{ + if (usbhid->urbout) + usbhid_restart_out_queue(usbhid); + usbhid_restart_ctrl_queue(usbhid); +} + +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_free_buffers(struct usb_device *dev, struct hid_device *hid) { struct usbhid_device *usbhid = hid->driver_data; @@ -868,11 +1027,11 @@ static struct hid_device *usb_hid_config init_waitqueue_head(&hid->wait); INIT_WORK(&usbhid->reset_work, hid_reset); + INIT_WORK(&usbhid->restart_work, __usbhid_restart_queues); setup_timer(&usbhid->io_retry, hid_retry_timeout, (unsigned long) hid); spin_lock_init(&usbhid->inlock); - spin_lock_init(&usbhid->outlock); - spin_lock_init(&usbhid->ctrllock); + spin_lock_init(&usbhid->outputlock); hid->version = le16_to_cpu(hdesc->bcdHID); hid->country = hdesc->bCountryCode; @@ -984,6 +1143,8 @@ static int hid_probe(struct usb_interfac if (!(hid = usb_hid_configure(intf))) return -ENODEV; + usb_set_intfdata(intf, hid); + usbhid_init_reports(hid); hid_dump_device(hid); if (hid->quirks & HID_QUIRK_RESET_LEDS) @@ -994,7 +1155,6 @@ static int hid_probe(struct usb_interfac 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"); @@ -1036,16 +1196,43 @@ static int hid_probe(struct usb_interfac return 0; } +static void hid_cease_io(struct usbhid_device *usbhid) +{ + del_timer(&usbhid->io_retry); + usb_kill_urb(usbhid->urbin); + usb_kill_urb(usbhid->urbctrl); + usb_kill_urb(usbhid->urbout); + flush_scheduled_work(); +} + static int hid_suspend(struct usb_interface *intf, pm_message_t message) { - struct hid_device *hid = usb_get_intfdata (intf); + struct hid_device *hid = usb_get_intfdata(intf); struct usbhid_device *usbhid = hid->driver_data; + struct usb_device *udev = interface_to_usbdev(intf); - spin_lock_irq(&usbhid->inlock); /* Sync with error handler */ - set_bit(HID_SUSPENDED, &usbhid->iofl); - spin_unlock_irq(&usbhid->inlock); - del_timer(&usbhid->io_retry); - usb_kill_urb(usbhid->urbin); + if (udev->auto_pm) { + spin_lock_irq(&usbhid->inlock); /* Sync with error handler */ + if (!test_bit(HID_RESET_PENDING, &usbhid->iofl) + && !test_bit(HID_CLEAR_HALT, &usbhid->iofl) + && !test_bit(HID_OUT_RUNNING, &usbhid->iofl) + && !test_bit(HID_CTRL_RUNNING, &usbhid->iofl)) + { + set_bit(HID_REPORTED_IDLE, &usbhid->iofl); + spin_unlock_irq(&usbhid->inlock); + } else { + spin_unlock_irq(&usbhid->inlock); + return -EBUSY; + } + } else { + spin_lock_irq(&usbhid->inlock); + set_bit(HID_REPORTED_IDLE, &usbhid->iofl); + spin_unlock_irq(&usbhid->inlock); + if (usbhid_wait_io(hid) < 0) + return -EIO; + } + + hid_cease_io(usbhid); dev_dbg(&intf->dev, "suspend\n"); return 0; } @@ -1056,28 +1243,63 @@ static int hid_resume(struct usb_interfa struct usbhid_device *usbhid = hid->driver_data; int status; - clear_bit(HID_SUSPENDED, &usbhid->iofl); + clear_bit(HID_REPORTED_IDLE, &usbhid->iofl); + usbhid_mark_busy(usbhid); usbhid->retry_delay = 0; + status = hid_start_in(hid); + if (status < 0) + hid_io_error(hid); + usbhid_restart_queues(usbhid); dev_dbg(&intf->dev, "resume status %d\n", status); - return status; + + return 0; } /* Treat USB reset pretty much the same as suspend/resume */ static void hid_pre_reset(struct usb_interface *intf) { - /* FIXME: What if the interface is already suspended? */ - hid_suspend(intf, PMSG_ON); + struct hid_device *hid = usb_get_intfdata(intf); + struct usbhid_device *usbhid = hid->driver_data; + + spin_lock_irq(&usbhid->inlock); + set_bit(HID_RESET_PENDING, &usbhid->iofl); + spin_unlock_irq(&usbhid->inlock); + hid_cease_io(usbhid); } static void hid_post_reset(struct usb_interface *intf) { - struct usb_device *dev = interface_to_usbdev (intf); + struct usb_device *dev = interface_to_usbdev(intf); + struct hid_device *hid = usb_get_intfdata(intf); + struct usbhid_device *usbhid = hid->driver_data; + int status; + + spin_lock_irq(&usbhid->inlock); + clear_bit(HID_RESET_PENDING, &usbhid->iofl); + spin_unlock_irq(&usbhid->inlock); hid_set_idle(dev, intf->cur_altsetting->desc.bInterfaceNumber, 0, 0); /* FIXME: Any more reinitialization needed? */ - hid_resume(intf); + status = hid_start_in(hid); + if (status < 0) + hid_io_error(hid); + usbhid_restart_queues(usbhid); +} + +int usbhid_get_power(struct hid_device *hid) +{ + struct usbhid_device *usbhid = hid->driver_data; + + return usb_autopm_get_interface(usbhid->intf); +} + +void usbhid_put_power(struct hid_device *hid) +{ + struct usbhid_device *usbhid = hid->driver_data; + + usb_autopm_put_interface(usbhid->intf); } static struct usb_device_id hid_usb_ids [] = { @@ -1097,6 +1319,7 @@ static struct usb_driver hid_driver = { .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.22-rc2/drivers/hid/usbhid/hiddev.c 2007-05-22 14:50:09.000000000 +0200 +++ linux-2.6.22-rc2-autosuspend/drivers/hid/usbhid/hiddev.c 2007-05-22 11:15:44.000000000 +0200 @@ -248,10 +248,12 @@ static int hiddev_release(struct inode * spin_unlock_irqrestore(&list->hiddev->list_lock, flags); if (!--list->hiddev->open) { - if (list->hiddev->exist) + if (list->hiddev->exist) { usbhid_close(list->hiddev->hid); - else + usbhid_put_power(list->hiddev->hid); + } else { kfree(list->hiddev); + } } kfree(list); @@ -266,6 +268,7 @@ static int hiddev_open(struct inode *ino { struct hiddev_list *list; unsigned long flags; + int res; int i = iminor(inode) - HIDDEV_MINOR_BASE; @@ -284,8 +287,13 @@ static int hiddev_open(struct inode *ino file->private_data = list; if (!list->hiddev->open++) - if (list->hiddev->exist) - usbhid_open(hiddev_table[i]->hid); + if (list->hiddev->exist) { + struct hid_device *hid = hiddev_table[i]->hid; + res = usbhid_get_power(hid); + if (res < 0) + return -EIO; + usbhid_open(hid); + } return 0; } ------------------------------------------------------------------------- This SF.net email is sponsored by DB2 Express Download DB2 Express C - the FREE version of DB2 express and take control of your XML. No limits. Just data. Click to get it now. http://sourceforge.net/powerbar/db2/ _______________________________________________ linux-usb-devel@lists.sourceforge.net To unsubscribe, use the last form field at: https://lists.sourceforge.net/lists/listinfo/linux-usb-devel