Commit ab05a2e4 authored by Johan Hovold's avatar Johan Hovold Committed by Ben Hutchings

USB: cdc-acm: fix broken runtime suspend

commit 140cb81a upstream.

The current ACM runtime-suspend implementation is broken in several
ways:

Firstly, it buffers only the first write request being made while
suspended -- any further writes are silently dropped.

Secondly, writes being dropped also leak write urbs, which are never
reclaimed (until the device is unbound).

Thirdly, even the single buffered write is not cleared at shutdown
(which may happen before the device is resumed), something which can
lead to another urb leak as well as a PM usage-counter leak.

Fix this by implementing a delayed-write queue using urb anchors and
making sure to discard the queue properly at shutdown.

Fixes: 11ea859d ("USB: additional power savings for cdc-acm devices
that support remote wakeup")
Reported-by: default avatarXiao Jin <jin.xiao@intel.com>
Signed-off-by: default avatarJohan Hovold <jhovold@gmail.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
[bwh: Backported to 3.2: adjust context]
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 16142775
...@@ -183,12 +183,9 @@ static int acm_write_start(struct acm *acm, int wbn) ...@@ -183,12 +183,9 @@ static int acm_write_start(struct acm *acm, int wbn)
acm->susp_count); acm->susp_count);
usb_autopm_get_interface_async(acm->control); usb_autopm_get_interface_async(acm->control);
if (acm->susp_count) { if (acm->susp_count) {
if (!acm->delayed_wb) usb_anchor_urb(wb->urb, &acm->delayed);
acm->delayed_wb = wb;
else
usb_autopm_put_interface_async(acm->control);
spin_unlock_irqrestore(&acm->write_lock, flags); spin_unlock_irqrestore(&acm->write_lock, flags);
return 0; /* A white lie */ return 0;
} }
usb_mark_last_busy(acm->dev); usb_mark_last_busy(acm->dev);
...@@ -545,11 +542,23 @@ static void acm_tty_unregister(struct acm *acm) ...@@ -545,11 +542,23 @@ static void acm_tty_unregister(struct acm *acm)
static void acm_port_down(struct acm *acm) static void acm_port_down(struct acm *acm)
{ {
struct urb *urb;
struct acm_wb *wb;
int i; int i;
if (acm->dev) { if (acm->dev) {
usb_autopm_get_interface(acm->control); usb_autopm_get_interface(acm->control);
acm_set_control(acm, acm->ctrlout = 0); acm_set_control(acm, acm->ctrlout = 0);
for (;;) {
urb = usb_get_from_anchor(&acm->delayed);
if (!urb)
break;
wb = urb->context;
wb->use = 0;
usb_autopm_put_interface_async(acm->control);
}
usb_kill_urb(acm->ctrlurb); usb_kill_urb(acm->ctrlurb);
for (i = 0; i < ACM_NW; i++) for (i = 0; i < ACM_NW; i++)
usb_kill_urb(acm->wb[i].urb); usb_kill_urb(acm->wb[i].urb);
...@@ -1112,6 +1121,7 @@ static int acm_probe(struct usb_interface *intf, ...@@ -1112,6 +1121,7 @@ static int acm_probe(struct usb_interface *intf,
acm->bInterval = epread->bInterval; acm->bInterval = epread->bInterval;
tty_port_init(&acm->port); tty_port_init(&acm->port);
acm->port.ops = &acm_port_ops; acm->port.ops = &acm_port_ops;
init_usb_anchor(&acm->delayed);
buf = usb_alloc_coherent(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma); buf = usb_alloc_coherent(usb_dev, ctrlsize, GFP_KERNEL, &acm->ctrl_dma);
if (!buf) { if (!buf) {
...@@ -1369,7 +1379,7 @@ static int acm_suspend(struct usb_interface *intf, pm_message_t message) ...@@ -1369,7 +1379,7 @@ static int acm_suspend(struct usb_interface *intf, pm_message_t message)
static int acm_resume(struct usb_interface *intf) static int acm_resume(struct usb_interface *intf)
{ {
struct acm *acm = usb_get_intfdata(intf); struct acm *acm = usb_get_intfdata(intf);
struct acm_wb *wb; struct urb *urb;
int rv = 0; int rv = 0;
mutex_lock(&acm->mutex); mutex_lock(&acm->mutex);
...@@ -1382,10 +1392,12 @@ static int acm_resume(struct usb_interface *intf) ...@@ -1382,10 +1392,12 @@ static int acm_resume(struct usb_interface *intf)
if (acm->port.count) { if (acm->port.count) {
rv = usb_submit_urb(acm->ctrlurb, GFP_ATOMIC); rv = usb_submit_urb(acm->ctrlurb, GFP_ATOMIC);
if (acm->delayed_wb) { for (;;) {
wb = acm->delayed_wb; urb = usb_get_from_anchor(&acm->delayed);
acm->delayed_wb = NULL; if (!urb)
acm_start_wb(acm, wb); break;
acm_start_wb(acm, urb->context);
} }
/* /*
......
...@@ -116,7 +116,7 @@ struct acm { ...@@ -116,7 +116,7 @@ struct acm {
unsigned int throttled:1; /* actually throttled */ unsigned int throttled:1; /* actually throttled */
unsigned int throttle_req:1; /* throttle requested */ unsigned int throttle_req:1; /* throttle requested */
u8 bInterval; u8 bInterval;
struct acm_wb *delayed_wb; /* write queued for a device about to be woken */ struct usb_anchor delayed; /* writes queued for a device about to be woken */
}; };
#define CDC_DATA_INTERFACE_TYPE 0x0a #define CDC_DATA_INTERFACE_TYPE 0x0a
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment