Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

USB: add NO_D3_DURING_SLEEP flag and revert 151b61284776be2

This patch (as1558) fixes a problem affecting several ASUS computers:
The machine crashes or corrupts memory when going into suspend if the
ehci-hcd driver is bound to any controllers. Users have been forced
to unbind or unload ehci-hcd before putting their systems to sleep.

After extensive testing, it was determined that the machines don't
like going into suspend when any EHCI controllers are in the PCI D3
power state. Presumably this is a firmware bug, but there's nothing
we can do about it except to avoid putting the controllers in D3
during system sleep.

The patch adds a new flag to indicate whether the problem is present,
and avoids changing the controller's power state if the flag is set.
Runtime suspend is unaffected; this matters only for system suspend.
However as a side effect, the controller will not respond to remote
wakeup requests while the system is asleep. Hence USB wakeup is not
functional -- but of course, this is already true in the current state
of affairs.

A similar patch has already been applied as commit
151b61284776be2d6f02d48c23c3625678960b97 (USB: EHCI: fix crash during
suspend on ASUS computers). The patch supersedes that one and reverts
it. There are two differences:

The old patch added the flag at the USB level; this patch
adds it at the PCI level.

The old patch applied to all chipsets with the same vendor,
subsystem vendor, and product IDs; this patch makes an
exception for a known-good system (based on DMI information).

Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Tested-by: Dâniel Fraga <fragabr@gmail.com>
Tested-by: Andrey Rahmatullin <wrar@wrar.name>
Tested-by: Steven Rostedt <rostedt@goodmis.org>
Cc: stable <stable@vger.kernel.org>
Reviewed-by: Rafael J. Wysocki <rjw@sisk.pl>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>

authored by

Alan Stern and committed by
Greg Kroah-Hartman
c2fb8a3f 0ef0be15

+33 -19
+5
drivers/pci/pci.c
··· 1744 1744 if (target_state == PCI_POWER_ERROR) 1745 1745 return -EIO; 1746 1746 1747 + /* Some devices mustn't be in D3 during system sleep */ 1748 + if (target_state == PCI_D3hot && 1749 + (dev->dev_flags & PCI_DEV_FLAGS_NO_D3_DURING_SLEEP)) 1750 + return 0; 1751 + 1747 1752 pci_enable_wake(dev, target_state, device_may_wakeup(&dev->dev)); 1748 1753 1749 1754 error = pci_set_power_state(dev, target_state);
+26
drivers/pci/quirks.c
··· 2929 2929 DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x0102, disable_igfx_irq); 2930 2930 DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x010a, disable_igfx_irq); 2931 2931 2932 + /* 2933 + * The Intel 6 Series/C200 Series chipset's EHCI controllers on many 2934 + * ASUS motherboards will cause memory corruption or a system crash 2935 + * if they are in D3 while the system is put into S3 sleep. 2936 + */ 2937 + static void __devinit asus_ehci_no_d3(struct pci_dev *dev) 2938 + { 2939 + const char *sys_info; 2940 + static const char good_Asus_board[] = "P8Z68-V"; 2941 + 2942 + if (dev->dev_flags & PCI_DEV_FLAGS_NO_D3_DURING_SLEEP) 2943 + return; 2944 + if (dev->subsystem_vendor != PCI_VENDOR_ID_ASUSTEK) 2945 + return; 2946 + sys_info = dmi_get_system_info(DMI_BOARD_NAME); 2947 + if (sys_info && memcmp(sys_info, good_Asus_board, 2948 + sizeof(good_Asus_board) - 1) == 0) 2949 + return; 2950 + 2951 + dev_info(&dev->dev, "broken D3 during system sleep on ASUS\n"); 2952 + dev->dev_flags |= PCI_DEV_FLAGS_NO_D3_DURING_SLEEP; 2953 + device_set_wakeup_capable(&dev->dev, false); 2954 + } 2955 + DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x1c26, asus_ehci_no_d3); 2956 + DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_INTEL, 0x1c2d, asus_ehci_no_d3); 2957 + 2932 2958 static void pci_do_fixups(struct pci_dev *dev, struct pci_fixup *f, 2933 2959 struct pci_fixup *end) 2934 2960 {
-9
drivers/usb/core/hcd-pci.c
··· 493 493 494 494 pci_save_state(pci_dev); 495 495 496 - /* 497 - * Some systems crash if an EHCI controller is in D3 during 498 - * a sleep transition. We have to leave such controllers in D0. 499 - */ 500 - if (hcd->broken_pci_sleep) { 501 - dev_dbg(dev, "Staying in PCI D0\n"); 502 - return retval; 503 - } 504 - 505 496 /* If the root hub is dead rather than suspended, disallow remote 506 497 * wakeup. usb_hc_died() should ensure that both hosts are marked as 507 498 * dying, so we only need to check the primary roothub.
-8
drivers/usb/host/ehci-pci.c
··· 144 144 hcd->has_tt = 1; 145 145 tdi_reset(ehci); 146 146 } 147 - if (pdev->subsystem_vendor == PCI_VENDOR_ID_ASUSTEK) { 148 - /* EHCI #1 or #2 on 6 Series/C200 Series chipset */ 149 - if (pdev->device == 0x1c26 || pdev->device == 0x1c2d) { 150 - ehci_info(ehci, "broken D3 during system sleep on ASUS\n"); 151 - hcd->broken_pci_sleep = 1; 152 - device_set_wakeup_capable(&pdev->dev, false); 153 - } 154 - } 155 147 break; 156 148 case PCI_VENDOR_ID_TDI: 157 149 if (pdev->device == PCI_DEVICE_ID_TDI_EHCI) {
+2
include/linux/pci.h
··· 176 176 PCI_DEV_FLAGS_NO_D3 = (__force pci_dev_flags_t) 2, 177 177 /* Provide indication device is assigned by a Virtual Machine Manager */ 178 178 PCI_DEV_FLAGS_ASSIGNED = (__force pci_dev_flags_t) 4, 179 + /* Device causes system crash if in D3 during S3 sleep */ 180 + PCI_DEV_FLAGS_NO_D3_DURING_SLEEP = (__force pci_dev_flags_t) 8, 179 181 }; 180 182 181 183 enum pci_irq_reroute_variant {
-2
include/linux/usb/hcd.h
··· 126 126 unsigned wireless:1; /* Wireless USB HCD */ 127 127 unsigned authorized_default:1; 128 128 unsigned has_tt:1; /* Integrated TT in root hub */ 129 - unsigned broken_pci_sleep:1; /* Don't put the 130 - controller in PCI-D3 for system sleep */ 131 129 132 130 unsigned int irq; /* irq allocated */ 133 131 void __iomem *regs; /* device memory/io */