aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb/class
diff options
context:
space:
mode:
authorJohan Hovold <jhovold@gmail.com>2014-05-26 13:23:45 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2014-05-27 18:04:09 -0400
commit5292afa657d0e790b7479ad8eef9450c1e040b3d (patch)
tree46a9bbf25625e66c77922f08e61ed68ebabc2374 /drivers/usb/class
parente4c36076c2a6195ec62c35b03c3fde84d0087dc8 (diff)
USB: cdc-acm: fix runtime PM imbalance at shutdown
Make sure only to decrement the PM counters if they were actually incremented. Note that the USB PM counter, but not necessarily the driver core PM counter, is reset when the interface is unbound. Fixes: 11ea859d64b6 ("USB: additional power savings for cdc-acm devices that support remote wakeup") Signed-off-by: Johan Hovold <jhovold@gmail.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb/class')
-rw-r--r--drivers/usb/class/cdc-acm.c6
1 files changed, 4 insertions, 2 deletions
diff --git a/drivers/usb/class/cdc-acm.c b/drivers/usb/class/cdc-acm.c
index eddeba61a88c..6bbd203f1861 100644
--- a/drivers/usb/class/cdc-acm.c
+++ b/drivers/usb/class/cdc-acm.c
@@ -590,12 +590,13 @@ static void acm_port_shutdown(struct tty_port *port)
590 struct urb *urb; 590 struct urb *urb;
591 struct acm_wb *wb; 591 struct acm_wb *wb;
592 int i; 592 int i;
593 int pm_err;
593 594
594 dev_dbg(&acm->control->dev, "%s\n", __func__); 595 dev_dbg(&acm->control->dev, "%s\n", __func__);
595 596
596 mutex_lock(&acm->mutex); 597 mutex_lock(&acm->mutex);
597 if (!acm->disconnected) { 598 if (!acm->disconnected) {
598 usb_autopm_get_interface(acm->control); 599 pm_err = usb_autopm_get_interface(acm->control);
599 acm_set_control(acm, acm->ctrlout = 0); 600 acm_set_control(acm, acm->ctrlout = 0);
600 601
601 for (;;) { 602 for (;;) {
@@ -613,7 +614,8 @@ static void acm_port_shutdown(struct tty_port *port)
613 for (i = 0; i < acm->rx_buflimit; i++) 614 for (i = 0; i < acm->rx_buflimit; i++)
614 usb_kill_urb(acm->read_urbs[i]); 615 usb_kill_urb(acm->read_urbs[i]);
615 acm->control->needs_remote_wakeup = 0; 616 acm->control->needs_remote_wakeup = 0;
616 usb_autopm_put_interface(acm->control); 617 if (!pm_err)
618 usb_autopm_put_interface(acm->control);
617 } 619 }
618 mutex_unlock(&acm->mutex); 620 mutex_unlock(&acm->mutex);
619} 621}