Hi,
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?
Regards
Oliver
PS: against vanilla 2.6.20
----
--- 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/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;
--- 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
14:06:14.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)
@@ -1308,6 +1529,7 @@
hid->claimed |= HID_CLAIMED_HIDDEV;
usb_set_intfdata(intf, hid);
+ intf->needs_remote_wakeup = 1;
if (!hid->claimed) {
printk ("HID device not claimed by input or hiddev\n");
@@ -1331,6 +1553,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 +1567,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 +1585,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 +1597,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 +1642,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/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) \
-------------------------------------------------------------------------
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
_______________________________________________
[email protected]
To unsubscribe, use the last form field at:
https://lists.sourceforge.net/lists/listinfo/linux-usb-devel