aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb/host
diff options
context:
space:
mode:
authorLan Tianyu <tianyu.lan@intel.com>2012-09-05 01:44:36 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2012-09-10 16:04:01 -0400
commitf7ac7787ad361e31a7972e2854ed8dc2eedfac3b (patch)
tree370b620b012a42f0890c829793dcc6f705c39cad /drivers/usb/host
parent693d8eb853f62a1341cf59df151b12c053f34e8a (diff)
usb/acpi: Use ACPI methods to power off ports.
Upcoming Intel systems will have an ACPI method to control whether a USB port can be completely powered off. The implication of powering off a USB port is that the device and host sees a physical disconnect, and subsequent port connections and remote wakeups will be lost. Add a new function, usb_acpi_power_manageable(), that can be used to find whether the usb port has ACPI power resources that can be used to power on and off the port on these machines. Also add a new function called usb_acpi_set_power_state() that controls the port power via these ACPI methods. When the USB core calls into the xHCI hub driver to power off a port, check whether the port can be completely powered off via this new ACPI mechanism. If so, call into these new ACPI methods. Also use the ACPI methods when the USB core asks to power on a port. Signed-off-by: Lan Tianyu <tianyu.lan@intel.com> Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb/host')
-rw-r--r--drivers/usb/host/xhci-hub.c12
1 files changed, 12 insertions, 0 deletions
diff --git a/drivers/usb/host/xhci-hub.c b/drivers/usb/host/xhci-hub.c
index 03032b3e6ed2..630e9e6e06b5 100644
--- a/drivers/usb/host/xhci-hub.c
+++ b/drivers/usb/host/xhci-hub.c
@@ -766,6 +766,12 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
766 766
767 temp = xhci_readl(xhci, port_array[wIndex]); 767 temp = xhci_readl(xhci, port_array[wIndex]);
768 xhci_dbg(xhci, "set port power, actual port %d status = 0x%x\n", wIndex, temp); 768 xhci_dbg(xhci, "set port power, actual port %d status = 0x%x\n", wIndex, temp);
769
770 temp = usb_acpi_power_manageable(hcd->self.root_hub,
771 wIndex);
772 if (temp)
773 usb_acpi_set_power_state(hcd->self.root_hub,
774 wIndex, true);
769 break; 775 break;
770 case USB_PORT_FEAT_RESET: 776 case USB_PORT_FEAT_RESET:
771 temp = (temp | PORT_RESET); 777 temp = (temp | PORT_RESET);
@@ -868,6 +874,12 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
868 case USB_PORT_FEAT_POWER: 874 case USB_PORT_FEAT_POWER:
869 xhci_writel(xhci, temp & ~PORT_POWER, 875 xhci_writel(xhci, temp & ~PORT_POWER,
870 port_array[wIndex]); 876 port_array[wIndex]);
877
878 temp = usb_acpi_power_manageable(hcd->self.root_hub,
879 wIndex);
880 if (temp)
881 usb_acpi_set_power_state(hcd->self.root_hub,
882 wIndex, false);
871 break; 883 break;
872 default: 884 default:
873 goto error; 885 goto error;