提交 f8aeb3bb 编写于 作者: D David Brownell 提交者: Greg Kroah-Hartman

[PATCH] USB: EHCI and NF2 quirk

This teaches the EHCI driver about a quirk seen in older NForce2 chips,
adding a workaround to ignore selective suspend requests.  Bus-wide
(so-called "global") suspend still works, as does USB wakeup of a
root hub that's globally suspended.

There's still a hole in this support though.  Strictly speaking, this
should _fail_ selective suspend requests, rather than ignoring them,
since doing it this way means that devices which should be able to issue
remote wakeup are not going to be able to do that.  For now, we'll just
live with that problem ... since usbcore expects to do selective suspend
on the way towards a full bus suspend, and usbcore needs to be able to
do full bus suspend.
Signed-off-by: NDavid Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 4186ecf8
...@@ -359,6 +359,8 @@ static int ehci_hub_control ( ...@@ -359,6 +359,8 @@ static int ehci_hub_control (
case USB_PORT_FEAT_SUSPEND: case USB_PORT_FEAT_SUSPEND:
if (temp & PORT_RESET) if (temp & PORT_RESET)
goto error; goto error;
if (ehci->no_selective_suspend)
break;
if (temp & PORT_SUSPEND) { if (temp & PORT_SUSPEND) {
if ((temp & PORT_PE) == 0) if ((temp & PORT_PE) == 0)
goto error; goto error;
...@@ -514,6 +516,8 @@ static int ehci_hub_control ( ...@@ -514,6 +516,8 @@ static int ehci_hub_control (
temp &= ~PORT_RWC_BITS; temp &= ~PORT_RWC_BITS;
switch (wValue) { switch (wValue) {
case USB_PORT_FEAT_SUSPEND: case USB_PORT_FEAT_SUSPEND:
if (ehci->no_selective_suspend)
break;
if ((temp & PORT_PE) == 0 if ((temp & PORT_PE) == 0
|| (temp & PORT_RESET) != 0) || (temp & PORT_RESET) != 0)
goto error; goto error;
......
...@@ -106,11 +106,11 @@ static int ehci_pci_setup(struct usb_hcd *hcd) ...@@ -106,11 +106,11 @@ static int ehci_pci_setup(struct usb_hcd *hcd)
} }
break; break;
case PCI_VENDOR_ID_NVIDIA: case PCI_VENDOR_ID_NVIDIA:
switch (pdev->device) {
/* NVidia reports that certain chips don't handle /* NVidia reports that certain chips don't handle
* QH, ITD, or SITD addresses above 2GB. (But TD, * QH, ITD, or SITD addresses above 2GB. (But TD,
* data buffer, and periodic schedule are normal.) * data buffer, and periodic schedule are normal.)
*/ */
switch (pdev->device) {
case 0x003c: /* MCP04 */ case 0x003c: /* MCP04 */
case 0x005b: /* CK804 */ case 0x005b: /* CK804 */
case 0x00d8: /* CK8 */ case 0x00d8: /* CK8 */
...@@ -120,6 +120,14 @@ static int ehci_pci_setup(struct usb_hcd *hcd) ...@@ -120,6 +120,14 @@ static int ehci_pci_setup(struct usb_hcd *hcd)
ehci_warn(ehci, "can't enable NVidia " ehci_warn(ehci, "can't enable NVidia "
"workaround for >2GB RAM\n"); "workaround for >2GB RAM\n");
break; break;
/* Some NForce2 chips have problems with selective suspend;
* fixed in newer silicon.
*/
case 0x0068:
pci_read_config_dword(pdev, PCI_REVISION_ID, &temp);
if ((temp & 0xff) < 0xa4)
ehci->no_selective_suspend = 1;
break;
} }
break; break;
} }
...@@ -163,6 +171,21 @@ static int ehci_pci_setup(struct usb_hcd *hcd) ...@@ -163,6 +171,21 @@ static int ehci_pci_setup(struct usb_hcd *hcd)
device_init_wakeup(&pdev->dev, 1); device_init_wakeup(&pdev->dev, 1);
} }
#ifdef CONFIG_USB_SUSPEND
/* REVISIT: the controller works fine for wakeup iff the root hub
* itself is "globally" suspended, but usbcore currently doesn't
* understand such things.
*
* System suspend currently expects to be able to suspend the entire
* device tree, device-at-a-time. If we failed selective suspend
* reports, system suspend would fail; so the root hub code must claim
* success. That's lying to usbcore, and it matters for for runtime
* PM scenarios with selective suspend and remote wakeup...
*/
if (ehci->no_selective_suspend && device_can_wakeup(&pdev->dev))
ehci_warn(ehci, "selective suspend/wakeup unavailable\n");
#endif
retval = ehci_pci_reinit(ehci, pdev); retval = ehci_pci_reinit(ehci, pdev);
done: done:
return retval; return retval;
......
...@@ -89,6 +89,8 @@ struct ehci_hcd { /* one per controller */ ...@@ -89,6 +89,8 @@ struct ehci_hcd { /* one per controller */
u32 command; u32 command;
unsigned is_tdi_rh_tt:1; /* TDI roothub with TT */ unsigned is_tdi_rh_tt:1; /* TDI roothub with TT */
unsigned no_selective_suspend:1;
u8 sbrn; /* packed release number */
/* irq statistics */ /* irq statistics */
#ifdef EHCI_STATS #ifdef EHCI_STATS
...@@ -97,7 +99,6 @@ struct ehci_hcd { /* one per controller */ ...@@ -97,7 +99,6 @@ struct ehci_hcd { /* one per controller */
#else #else
# define COUNT(x) do {} while (0) # define COUNT(x) do {} while (0)
#endif #endif
u8 sbrn; /* packed release number */
}; };
/* convert between an HCD pointer and the corresponding EHCI_HCD */ /* convert between an HCD pointer and the corresponding EHCI_HCD */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册