aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb/host/ehci-hub.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/usb/host/ehci-hub.c')
-rw-r--r--drivers/usb/host/ehci-hub.c64
1 files changed, 35 insertions, 29 deletions
diff --git a/drivers/usb/host/ehci-hub.c b/drivers/usb/host/ehci-hub.c
index 12f881ff4b23..076474d95dbf 100644
--- a/drivers/usb/host/ehci-hub.c
+++ b/drivers/usb/host/ehci-hub.c
@@ -270,16 +270,14 @@ ehci_hub_status_data (struct usb_hcd *hcd, char *buf)
270 spin_lock_irqsave (&ehci->lock, flags); 270 spin_lock_irqsave (&ehci->lock, flags);
271 for (i = 0; i < ports; i++) { 271 for (i = 0; i < ports; i++) {
272 temp = ehci_readl(ehci, &ehci->regs->port_status [i]); 272 temp = ehci_readl(ehci, &ehci->regs->port_status [i]);
273 if (temp & PORT_OWNER) { 273
274 /* don't report this in GetPortStatus */ 274 /*
275 if (temp & PORT_CSC) { 275 * Return status information even for ports with OWNER set.
276 temp &= ~PORT_RWC_BITS; 276 * Otherwise khubd wouldn't see the disconnect event when a
277 temp |= PORT_CSC; 277 * high-speed device is switched over to the companion
278 ehci_writel(ehci, temp, 278 * controller by the user.
279 &ehci->regs->port_status [i]); 279 */
280 } 280
281 continue;
282 }
283 if (!(temp & PORT_CONNECT)) 281 if (!(temp & PORT_CONNECT))
284 ehci->reset_done [i] = 0; 282 ehci->reset_done [i] = 0;
285 if ((temp & mask) != 0 283 if ((temp & mask) != 0
@@ -377,8 +375,13 @@ static int ehci_hub_control (
377 goto error; 375 goto error;
378 wIndex--; 376 wIndex--;
379 temp = ehci_readl(ehci, status_reg); 377 temp = ehci_readl(ehci, status_reg);
380 if (temp & PORT_OWNER) 378
381 break; 379 /*
380 * Even if OWNER is set, so the port is owned by the
381 * companion controller, khubd needs to be able to clear
382 * the port-change status bits (especially
383 * USB_PORT_FEAT_C_CONNECTION).
384 */
382 385
383 switch (wValue) { 386 switch (wValue) {
384 case USB_PORT_FEAT_ENABLE: 387 case USB_PORT_FEAT_ENABLE:
@@ -501,24 +504,27 @@ static int ehci_hub_control (
501 ehci_readl(ehci, status_reg)); 504 ehci_readl(ehci, status_reg));
502 } 505 }
503 506
504 // don't show wPortStatus if it's owned by a companion hc 507 /*
505 if (!(temp & PORT_OWNER)) { 508 * Even if OWNER is set, there's no harm letting khubd
506 if (temp & PORT_CONNECT) { 509 * see the wPortStatus values (they should all be 0 except
507 status |= 1 << USB_PORT_FEAT_CONNECTION; 510 * for PORT_POWER anyway).
508 // status may be from integrated TT 511 */
509 status |= ehci_port_speed(ehci, temp); 512
510 } 513 if (temp & PORT_CONNECT) {
511 if (temp & PORT_PE) 514 status |= 1 << USB_PORT_FEAT_CONNECTION;
512 status |= 1 << USB_PORT_FEAT_ENABLE; 515 // status may be from integrated TT
513 if (temp & (PORT_SUSPEND|PORT_RESUME)) 516 status |= ehci_port_speed(ehci, temp);
514 status |= 1 << USB_PORT_FEAT_SUSPEND;
515 if (temp & PORT_OC)
516 status |= 1 << USB_PORT_FEAT_OVER_CURRENT;
517 if (temp & PORT_RESET)
518 status |= 1 << USB_PORT_FEAT_RESET;
519 if (temp & PORT_POWER)
520 status |= 1 << USB_PORT_FEAT_POWER;
521 } 517 }
518 if (temp & PORT_PE)
519 status |= 1 << USB_PORT_FEAT_ENABLE;
520 if (temp & (PORT_SUSPEND|PORT_RESUME))
521 status |= 1 << USB_PORT_FEAT_SUSPEND;
522 if (temp & PORT_OC)
523 status |= 1 << USB_PORT_FEAT_OVER_CURRENT;
524 if (temp & PORT_RESET)
525 status |= 1 << USB_PORT_FEAT_RESET;
526 if (temp & PORT_POWER)
527 status |= 1 << USB_PORT_FEAT_POWER;
522 528
523#ifndef EHCI_VERBOSE_DEBUG 529#ifndef EHCI_VERBOSE_DEBUG
524 if (status & ~0xffff) /* only if wPortChange is interesting */ 530 if (status & ~0xffff) /* only if wPortChange is interesting */