2  * Copyright (C) 2001-2004 by David Brownell
 
   4  * This program is free software; you can redistribute it and/or modify it
 
   5  * under the terms of the GNU General Public License as published by the
 
   6  * Free Software Foundation; either version 2 of the License, or (at your
 
   7  * option) any later version.
 
   9  * This program is distributed in the hope that it will be useful, but
 
  10  * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 
  11  * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 
  14  * You should have received a copy of the GNU General Public License
 
  15  * along with this program; if not, write to the Free Software Foundation,
 
  16  * Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 
  19 /* this file is part of ehci-hcd.c */
 
  21 /*-------------------------------------------------------------------------*/
 
  24  * EHCI Root Hub ... the nonsharable stuff
 
  26  * Registers don't need cpu_to_le32, that happens transparently
 
  29 /*-------------------------------------------------------------------------*/
 
  33 static int ehci_bus_suspend (struct usb_hcd *hcd)
 
  35         struct ehci_hcd         *ehci = hcd_to_ehci (hcd);
 
  38         if (time_before (jiffies, ehci->next_statechange))
 
  41         port = HCS_N_PORTS (ehci->hcs_params);
 
  42         spin_lock_irq (&ehci->lock);
 
  44         /* stop schedules, clean any completed work */
 
  45         if (HC_IS_RUNNING(hcd->state)) {
 
  47                 hcd->state = HC_STATE_QUIESCING;
 
  49         ehci->command = readl (&ehci->regs->command);
 
  51                 ehci->reclaim_ready = 1;
 
  52         ehci_work(ehci, NULL);
 
  54         /* suspend any active/unsuspended ports, maybe allow wakeup */
 
  56                 u32 __iomem     *reg = &ehci->regs->port_status [port];
 
  57                 u32             t1 = readl (reg) & ~PORT_RWC_BITS;
 
  60                 if ((t1 & PORT_PE) && !(t1 & PORT_OWNER))
 
  62                 if (device_may_wakeup(&hcd->self.root_hub->dev))
 
  63                         t2 |= PORT_WKOC_E|PORT_WKDISC_E|PORT_WKCONN_E;
 
  65                         t2 &= ~(PORT_WKOC_E|PORT_WKDISC_E|PORT_WKCONN_E);
 
  68                         ehci_vdbg (ehci, "port %d, %08x -> %08x\n",
 
  74         /* turn off now-idle HC */
 
  75         del_timer_sync (&ehci->watchdog);
 
  77         hcd->state = HC_STATE_SUSPENDED;
 
  79         ehci->next_statechange = jiffies + msecs_to_jiffies(10);
 
  80         spin_unlock_irq (&ehci->lock);
 
  85 /* caller has locked the root hub, and should reset/reinit on error */
 
  86 static int ehci_bus_resume (struct usb_hcd *hcd)
 
  88         struct ehci_hcd         *ehci = hcd_to_ehci (hcd);
 
  93         if (time_before (jiffies, ehci->next_statechange))
 
  95         spin_lock_irq (&ehci->lock);
 
  97         /* Ideally and we've got a real resume here, and no port's power
 
  98          * was lost.  (For PCI, that means Vaux was maintained.)  But we
 
  99          * could instead be restoring a swsusp snapshot -- so that BIOS was
 
 100          * the last user of the controller, not reset/pm hardware keeping
 
 101          * state we gave to it.
 
 104         /* re-init operational registers in case we lost power */
 
 105         if (readl (&ehci->regs->intr_enable) == 0) {
 
 106                 /* at least some APM implementations will try to deliver
 
 107                  * IRQs right away, so delay them until we're ready.
 
 110                 writel (0, &ehci->regs->segment);
 
 111                 writel (ehci->periodic_dma, &ehci->regs->frame_list);
 
 112                 writel ((u32)ehci->async->qh_dma, &ehci->regs->async_next);
 
 115         ehci_dbg(ehci, "resume root hub%s\n",
 
 116                         intr_enable ? " after power loss" : "");
 
 118         /* restore CMD_RUN, framelist size, and irq threshold */
 
 119         writel (ehci->command, &ehci->regs->command);
 
 121         /* take ports out of suspend */
 
 122         i = HCS_N_PORTS (ehci->hcs_params);
 
 124                 temp = readl (&ehci->regs->port_status [i]);
 
 125                 temp &= ~(PORT_RWC_BITS
 
 126                         | PORT_WKOC_E | PORT_WKDISC_E | PORT_WKCONN_E);
 
 127                 if (temp & PORT_SUSPEND) {
 
 128                         ehci->reset_done [i] = jiffies + msecs_to_jiffies (20);
 
 131                 writel (temp, &ehci->regs->port_status [i]);
 
 133         i = HCS_N_PORTS (ehci->hcs_params);
 
 136                 temp = readl (&ehci->regs->port_status [i]);
 
 137                 if ((temp & PORT_SUSPEND) == 0)
 
 139                 temp &= ~(PORT_RWC_BITS | PORT_RESUME);
 
 140                 writel (temp, &ehci->regs->port_status [i]);
 
 141                 ehci_vdbg (ehci, "resumed port %d\n", i + 1);
 
 143         (void) readl (&ehci->regs->command);
 
 145         /* maybe re-activate the schedule(s) */
 
 147         if (ehci->async->qh_next.qh)
 
 149         if (ehci->periodic_sched)
 
 152                 ehci->command |= temp;
 
 153                 writel (ehci->command, &ehci->regs->command);
 
 156         ehci->next_statechange = jiffies + msecs_to_jiffies(5);
 
 157         hcd->state = HC_STATE_RUNNING;
 
 159         /* Now we can safely re-enable irqs */
 
 161                 writel (INTR_MASK, &ehci->regs->intr_enable);
 
 163         spin_unlock_irq (&ehci->lock);
 
 169 #define ehci_bus_suspend        NULL
 
 170 #define ehci_bus_resume         NULL
 
 172 #endif  /* CONFIG_PM */
 
 174 /*-------------------------------------------------------------------------*/
 
 176 static int check_reset_complete (
 
 177         struct ehci_hcd *ehci,
 
 181         if (!(port_status & PORT_CONNECT)) {
 
 182                 ehci->reset_done [index] = 0;
 
 186         /* if reset finished and it's still not enabled -- handoff */
 
 187         if (!(port_status & PORT_PE)) {
 
 189                 /* with integrated TT, there's nobody to hand it to! */
 
 190                 if (ehci_is_TDI(ehci)) {
 
 192                                 "Failed to enable port %d on root hub TT\n",
 
 197                 ehci_dbg (ehci, "port %d full speed --> companion\n",
 
 200                 // what happens if HCS_N_CC(params) == 0 ?
 
 201                 port_status |= PORT_OWNER;
 
 202                 port_status &= ~PORT_RWC_BITS;
 
 203                 writel (port_status, &ehci->regs->port_status [index]);
 
 206                 ehci_dbg (ehci, "port %d high speed\n", index + 1);
 
 211 /*-------------------------------------------------------------------------*/
 
 214 /* build "status change" packet (one or two bytes) from HC registers */
 
 217 ehci_hub_status_data (struct usb_hcd *hcd, char *buf)
 
 219         struct ehci_hcd *ehci = hcd_to_ehci (hcd);
 
 220         u32             temp, status = 0;
 
 221         int             ports, i, retval = 1;
 
 224         /* if !USB_SUSPEND, root hub timers won't get shut down ... */
 
 225         if (!HC_IS_RUNNING(hcd->state))
 
 228         /* init status to no-changes */
 
 230         ports = HCS_N_PORTS (ehci->hcs_params);
 
 236         /* no hub change reports (bit 0) for now (power, ...) */
 
 238         /* port N changes (bit N)? */
 
 239         spin_lock_irqsave (&ehci->lock, flags);
 
 240         for (i = 0; i < ports; i++) {
 
 241                 temp = readl (&ehci->regs->port_status [i]);
 
 242                 if (temp & PORT_OWNER) {
 
 243                         /* don't report this in GetPortStatus */
 
 244                         if (temp & PORT_CSC) {
 
 245                                 temp &= ~PORT_RWC_BITS;
 
 247                                 writel (temp, &ehci->regs->port_status [i]);
 
 251                 if (!(temp & PORT_CONNECT))
 
 252                         ehci->reset_done [i] = 0;
 
 253                 if ((temp & (PORT_CSC | PORT_PEC | PORT_OCC)) != 0
 
 254                                 // PORT_STAT_C_SUSPEND?
 
 255                                 || ((temp & PORT_RESUME) != 0
 
 256                                         && time_after (jiffies,
 
 257                                                 ehci->reset_done [i]))) {
 
 259                             buf [0] |= 1 << (i + 1);
 
 261                             buf [1] |= 1 << (i - 7);
 
 265         /* FIXME autosuspend idle root hubs */
 
 266         spin_unlock_irqrestore (&ehci->lock, flags);
 
 267         return status ? retval : 0;
 
 270 /*-------------------------------------------------------------------------*/
 
 273 ehci_hub_descriptor (
 
 274         struct ehci_hcd                 *ehci,
 
 275         struct usb_hub_descriptor       *desc
 
 277         int             ports = HCS_N_PORTS (ehci->hcs_params);
 
 280         desc->bDescriptorType = 0x29;
 
 281         desc->bPwrOn2PwrGood = 10;      /* ehci 1.0, 2.3.9 says 20ms max */
 
 282         desc->bHubContrCurrent = 0;
 
 284         desc->bNbrPorts = ports;
 
 285         temp = 1 + (ports / 8);
 
 286         desc->bDescLength = 7 + 2 * temp;
 
 288         /* two bitmaps:  ports removable, and usb 1.0 legacy PortPwrCtrlMask */
 
 289         memset (&desc->bitmap [0], 0, temp);
 
 290         memset (&desc->bitmap [temp], 0xff, temp);
 
 292         temp = 0x0008;                  /* per-port overcurrent reporting */
 
 293         if (HCS_PPC (ehci->hcs_params))
 
 294                 temp |= 0x0001;         /* per-port power control */
 
 296                 temp |= 0x0002;         /* no power switching */
 
 298 // re-enable when we support USB_PORT_FEAT_INDICATOR below.
 
 299         if (HCS_INDICATOR (ehci->hcs_params))
 
 300                 temp |= 0x0080;         /* per-port indicators (LEDs) */
 
 302         desc->wHubCharacteristics = (__force __u16)cpu_to_le16 (temp);
 
 305 /*-------------------------------------------------------------------------*/
 
 307 #define PORT_WAKE_BITS  (PORT_WKOC_E|PORT_WKDISC_E|PORT_WKCONN_E)
 
 309 static int ehci_hub_control (
 
 317         struct ehci_hcd *ehci = hcd_to_ehci (hcd);
 
 318         int             ports = HCS_N_PORTS (ehci->hcs_params);
 
 324          * FIXME:  support SetPortFeatures USB_PORT_FEAT_INDICATOR.
 
 325          * HCS_INDICATOR may say we can change LEDs to off/amber/green.
 
 326          * (track current state ourselves) ... blink for diagnostics,
 
 327          * power, "this is the one", etc.  EHCI spec supports this.
 
 330         spin_lock_irqsave (&ehci->lock, flags);
 
 332         case ClearHubFeature:
 
 334                 case C_HUB_LOCAL_POWER:
 
 335                 case C_HUB_OVER_CURRENT:
 
 336                         /* no hub-wide feature/status flags */
 
 342         case ClearPortFeature:
 
 343                 if (!wIndex || wIndex > ports)
 
 346                 temp = readl (&ehci->regs->port_status [wIndex]);
 
 347                 if (temp & PORT_OWNER)
 
 351                 case USB_PORT_FEAT_ENABLE:
 
 352                         writel (temp & ~PORT_PE,
 
 353                                 &ehci->regs->port_status [wIndex]);
 
 355                 case USB_PORT_FEAT_C_ENABLE:
 
 356                         writel((temp & ~PORT_RWC_BITS) | PORT_PEC,
 
 357                                 &ehci->regs->port_status [wIndex]);
 
 359                 case USB_PORT_FEAT_SUSPEND:
 
 360                         if (temp & PORT_RESET)
 
 362                         if (ehci->no_selective_suspend)
 
 364                         if (temp & PORT_SUSPEND) {
 
 365                                 if ((temp & PORT_PE) == 0)
 
 367                                 /* resume signaling for 20 msec */
 
 368                                 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS);
 
 369                                 writel (temp | PORT_RESUME,
 
 370                                         &ehci->regs->port_status [wIndex]);
 
 371                                 ehci->reset_done [wIndex] = jiffies
 
 372                                                 + msecs_to_jiffies (20);
 
 375                 case USB_PORT_FEAT_C_SUSPEND:
 
 376                         /* we auto-clear this feature */
 
 378                 case USB_PORT_FEAT_POWER:
 
 379                         if (HCS_PPC (ehci->hcs_params))
 
 380                                 writel (temp & ~(PORT_RWC_BITS | PORT_POWER),
 
 381                                         &ehci->regs->port_status [wIndex]);
 
 383                 case USB_PORT_FEAT_C_CONNECTION:
 
 384                         writel((temp & ~PORT_RWC_BITS) | PORT_CSC,
 
 385                                 &ehci->regs->port_status [wIndex]);
 
 387                 case USB_PORT_FEAT_C_OVER_CURRENT:
 
 388                         writel((temp & ~PORT_RWC_BITS) | PORT_OCC,
 
 389                                 &ehci->regs->port_status [wIndex]);
 
 391                 case USB_PORT_FEAT_C_RESET:
 
 392                         /* GetPortStatus clears reset */
 
 397                 readl (&ehci->regs->command);   /* unblock posted write */
 
 399         case GetHubDescriptor:
 
 400                 ehci_hub_descriptor (ehci, (struct usb_hub_descriptor *)
 
 404                 /* no hub-wide feature/status flags */
 
 406                 //cpu_to_le32s ((u32 *) buf);
 
 409                 if (!wIndex || wIndex > ports)
 
 413                 temp = readl (&ehci->regs->port_status [wIndex]);
 
 417                         status |= 1 << USB_PORT_FEAT_C_CONNECTION;
 
 419                         status |= 1 << USB_PORT_FEAT_C_ENABLE;
 
 421                         status |= 1 << USB_PORT_FEAT_C_OVER_CURRENT;
 
 423                 /* whoever resumes must GetPortStatus to complete it!! */
 
 424                 if ((temp & PORT_RESUME)
 
 425                                 && time_after (jiffies,
 
 426                                         ehci->reset_done [wIndex])) {
 
 427                         status |= 1 << USB_PORT_FEAT_C_SUSPEND;
 
 428                         ehci->reset_done [wIndex] = 0;
 
 430                         /* stop resume signaling */
 
 431                         temp = readl (&ehci->regs->port_status [wIndex]);
 
 432                         writel (temp & ~(PORT_RWC_BITS | PORT_RESUME),
 
 433                                 &ehci->regs->port_status [wIndex]);
 
 435                                         &ehci->regs->port_status [wIndex],
 
 436                                         PORT_RESUME, 0, 2000 /* 2msec */);
 
 438                                 ehci_err (ehci, "port %d resume error %d\n",
 
 442                         temp &= ~(PORT_SUSPEND|PORT_RESUME|(3<<10));
 
 445                 /* whoever resets must GetPortStatus to complete it!! */
 
 446                 if ((temp & PORT_RESET)
 
 447                                 && time_after (jiffies,
 
 448                                         ehci->reset_done [wIndex])) {
 
 449                         status |= 1 << USB_PORT_FEAT_C_RESET;
 
 450                         ehci->reset_done [wIndex] = 0;
 
 452                         /* force reset to complete */
 
 453                         writel (temp & ~(PORT_RWC_BITS | PORT_RESET),
 
 454                                         &ehci->regs->port_status [wIndex]);
 
 455                         /* REVISIT:  some hardware needs 550+ usec to clear
 
 456                          * this bit; seems too long to spin routinely...
 
 459                                         &ehci->regs->port_status [wIndex],
 
 462                                 ehci_err (ehci, "port %d reset error %d\n",
 
 467                         /* see what we found out */
 
 468                         temp = check_reset_complete (ehci, wIndex,
 
 469                                 readl (&ehci->regs->port_status [wIndex]));
 
 472                 // don't show wPortStatus if it's owned by a companion hc
 
 473                 if (!(temp & PORT_OWNER)) {
 
 474                         if (temp & PORT_CONNECT) {
 
 475                                 status |= 1 << USB_PORT_FEAT_CONNECTION;
 
 476                                 // status may be from integrated TT
 
 477                                 status |= ehci_port_speed(ehci, temp);
 
 480                                 status |= 1 << USB_PORT_FEAT_ENABLE;
 
 481                         if (temp & (PORT_SUSPEND|PORT_RESUME))
 
 482                                 status |= 1 << USB_PORT_FEAT_SUSPEND;
 
 484                                 status |= 1 << USB_PORT_FEAT_OVER_CURRENT;
 
 485                         if (temp & PORT_RESET)
 
 486                                 status |= 1 << USB_PORT_FEAT_RESET;
 
 487                         if (temp & PORT_POWER)
 
 488                                 status |= 1 << USB_PORT_FEAT_POWER;
 
 491 #ifndef EHCI_VERBOSE_DEBUG
 
 492         if (status & ~0xffff)   /* only if wPortChange is interesting */
 
 494                 dbg_port (ehci, "GetStatus", wIndex + 1, temp);
 
 495                 // we "know" this alignment is good, caller used kmalloc()...
 
 496                 *((__le32 *) buf) = cpu_to_le32 (status);
 
 500                 case C_HUB_LOCAL_POWER:
 
 501                 case C_HUB_OVER_CURRENT:
 
 502                         /* no hub-wide feature/status flags */
 
 509                 if (!wIndex || wIndex > ports)
 
 512                 temp = readl (&ehci->regs->port_status [wIndex]);
 
 513                 if (temp & PORT_OWNER)
 
 516                 temp &= ~PORT_RWC_BITS;
 
 518                 case USB_PORT_FEAT_SUSPEND:
 
 519                         if (ehci->no_selective_suspend)
 
 521                         if ((temp & PORT_PE) == 0
 
 522                                         || (temp & PORT_RESET) != 0)
 
 524                         if (device_may_wakeup(&hcd->self.root_hub->dev))
 
 525                                 temp |= PORT_WAKE_BITS;
 
 526                         writel (temp | PORT_SUSPEND,
 
 527                                 &ehci->regs->port_status [wIndex]);
 
 529                 case USB_PORT_FEAT_POWER:
 
 530                         if (HCS_PPC (ehci->hcs_params))
 
 531                                 writel (temp | PORT_POWER,
 
 532                                         &ehci->regs->port_status [wIndex]);
 
 534                 case USB_PORT_FEAT_RESET:
 
 535                         if (temp & PORT_RESUME)
 
 537                         /* line status bits may report this as low speed,
 
 538                          * which can be fine if this root hub has a
 
 539                          * transaction translator built in.
 
 541                         if ((temp & (PORT_PE|PORT_CONNECT)) == PORT_CONNECT
 
 542                                         && !ehci_is_TDI(ehci)
 
 543                                         && PORT_USB11 (temp)) {
 
 545                                         "port %d low speed --> companion\n",
 
 549                                 ehci_vdbg (ehci, "port %d reset\n", wIndex + 1);
 
 554                                  * caller must wait, then call GetPortStatus
 
 555                                  * usb 2.0 spec says 50 ms resets on root
 
 557                                 ehci->reset_done [wIndex] = jiffies
 
 558                                                 + msecs_to_jiffies (50);
 
 560                         writel (temp, &ehci->regs->port_status [wIndex]);
 
 565                 readl (&ehci->regs->command);   /* unblock posted writes */
 
 570                 /* "stall" on error */
 
 573         spin_unlock_irqrestore (&ehci->lock, flags);