aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb/host/ehci-hub.c
diff options
context:
space:
mode:
authorAlek Du <alek.du@intel.com>2010-05-09 23:17:49 -0400
committerGreg Kroah-Hartman <gregkh@suse.de>2010-05-20 16:21:45 -0400
commiteab80de01cb398419ef3305f35abcb367c647c8b (patch)
tree0cbf986a987df044234bba1ad0a1c6565268a4f1 /drivers/usb/host/ehci-hub.c
parent17b2765ef8db8dc00a24a9afd4646bb3decffbd2 (diff)
USB: EHCI: clear PHCD before resuming
This is a bug fix for PHCD (phy clock disable) low power feature: After PHCD is set, any write to PORTSC register is illegal, so when resume ports, clear PHCD bit first. Signed-off-by: Alek Du <alek.du@intel.com> Cc: David Brownell <dbrownell@users.sourceforge.net> Cc: Alan Stern <stern@rowland.harvard.edu> Cc: stable <stable@kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'drivers/usb/host/ehci-hub.c')
-rw-r--r--drivers/usb/host/ehci-hub.c17
1 files changed, 17 insertions, 0 deletions
diff --git a/drivers/usb/host/ehci-hub.c b/drivers/usb/host/ehci-hub.c
index c44018109a13..ef956220f854 100644
--- a/drivers/usb/host/ehci-hub.c
+++ b/drivers/usb/host/ehci-hub.c
@@ -294,6 +294,16 @@ static int ehci_bus_resume (struct usb_hcd *hcd)
294 /* manually resume the ports we suspended during bus_suspend() */ 294 /* manually resume the ports we suspended during bus_suspend() */
295 i = HCS_N_PORTS (ehci->hcs_params); 295 i = HCS_N_PORTS (ehci->hcs_params);
296 while (i--) { 296 while (i--) {
297 /* clear phy low power mode before resume */
298 if (ehci->has_hostpc) {
299 u32 __iomem *hostpc_reg =
300 (u32 __iomem *)((u8 *)ehci->regs
301 + HOSTPC0 + 4 * (i & 0xff));
302 temp = ehci_readl(ehci, hostpc_reg);
303 ehci_writel(ehci, temp & ~HOSTPC_PHCD,
304 hostpc_reg);
305 mdelay(5);
306 }
297 temp = ehci_readl(ehci, &ehci->regs->port_status [i]); 307 temp = ehci_readl(ehci, &ehci->regs->port_status [i]);
298 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS); 308 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS);
299 if (test_bit(i, &ehci->bus_suspended) && 309 if (test_bit(i, &ehci->bus_suspended) &&
@@ -678,6 +688,13 @@ static int ehci_hub_control (
678 if (temp & PORT_SUSPEND) { 688 if (temp & PORT_SUSPEND) {
679 if ((temp & PORT_PE) == 0) 689 if ((temp & PORT_PE) == 0)
680 goto error; 690 goto error;
691 /* clear phy low power mode before resume */
692 if (hostpc_reg) {
693 temp1 = ehci_readl(ehci, hostpc_reg);
694 ehci_writel(ehci, temp1 & ~HOSTPC_PHCD,
695 hostpc_reg);
696 mdelay(5);
697 }
681 /* resume signaling for 20 msec */ 698 /* resume signaling for 20 msec */
682 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS); 699 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS);
683 ehci_writel(ehci, temp | PORT_RESUME, 700 ehci_writel(ehci, temp | PORT_RESUME,