intel/lynxpoint: Work around XHCI resume issues
When USB3 devices are attached while in suspend, or two USB3 devices that are both plugged in are switched to the other port while in suspend the kernel does not seem to notice this -- despite the cold attach status bit. This results in the devices showing up in the USB list at the old enumerated device numbers and higher layers continuing to think they are present but not reseponding. With the kernel workaround to deal with devices that are logically disconnected it is possible for firmware to send a warm port reset to devices that are in this state and then the kernel will see them disappear and handle it properly. This same issue exists in the EFI firmware on the Whitetip Mountain 2 reference board so it is not specifically a coreboot bug. If this behavior is fixed in the kernel then this workaround could be removed since it is in RW firmware. BUG=chrome-os-partner:22818 BRANCH=falco,peppy,wolf,leon TEST=manual: 1) attach two USB3 devices 2) suspend system 3) switch the ports that the USB3 devices are attatched to 4) resume system 5) confirm that the devices are re-enumerated and come up properly Original-Change-Id: Ifba3ffc94a06dc0b2436d7d7d464d824657362af Signed-off-by: Duncan Laurie <dlaurie@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/170335 Reviewed-by: Aaron Durbin <adurbin@chromium.org> (cherry picked from commit 203d200268f4af6445224962190cbc66ad2a83e4) Change-Id: I54fd2847ee25a60f25c2cefebdc1a3c18455464a Signed-off-by: Duncan Laurie <dlaurie@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/170579 [pm: rebase to master branch of coreboot upstream] Signed-off-by: Paul Menzel <paulepanter@users.sourceforge.net> Reviewed-on: http://review.coreboot.org/6017 Tested-by: build bot (Jenkins) Reviewed-by: Patrick Georgi <patrick@georgi-clan.de>
This commit is contained in:
parent
16a0f5c4e3
commit
32d2e2b360
1 changed files with 10 additions and 3 deletions
|
@ -37,8 +37,6 @@ static u32 usb_xhci_mem_base(device_t dev)
|
||||||
return mem_base & ~0xf;
|
return mem_base & ~0xf;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef __SMM__
|
|
||||||
|
|
||||||
static int usb_xhci_port_count_usb3(device_t dev)
|
static int usb_xhci_port_count_usb3(device_t dev)
|
||||||
{
|
{
|
||||||
if (pch_is_lp()) {
|
if (pch_is_lp()) {
|
||||||
|
@ -127,7 +125,8 @@ static void usb_xhci_reset_usb3(device_t dev, int all)
|
||||||
continue;
|
continue;
|
||||||
status = read32(portsc) & XHCI_USB3_PORTSC_PLS;
|
status = read32(portsc) & XHCI_USB3_PORTSC_PLS;
|
||||||
/* Reset all or only disconnected ports */
|
/* Reset all or only disconnected ports */
|
||||||
if (all || status == XHCI_PLSR_RXDETECT)
|
if (all || (status == XHCI_PLSR_RXDETECT ||
|
||||||
|
status == XHCI_PLSR_POLLING))
|
||||||
usb_xhci_reset_port_usb3(mem_base, port);
|
usb_xhci_reset_port_usb3(mem_base, port);
|
||||||
else
|
else
|
||||||
port_disabled |= 1 << port;
|
port_disabled |= 1 << port;
|
||||||
|
@ -156,6 +155,8 @@ static void usb_xhci_reset_usb3(device_t dev, int all)
|
||||||
usb_xhci_reset_status_usb3(mem_base, port);
|
usb_xhci_reset_status_usb3(mem_base, port);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef __SMM__
|
||||||
|
|
||||||
/* Handler for XHCI controller on entry to S3/S4/S5 */
|
/* Handler for XHCI controller on entry to S3/S4/S5 */
|
||||||
void usb_xhci_sleep_prepare(device_t dev, u8 slp_typ)
|
void usb_xhci_sleep_prepare(device_t dev, u8 slp_typ)
|
||||||
{
|
{
|
||||||
|
@ -350,6 +351,12 @@ static void usb_xhci_init(device_t dev)
|
||||||
reg32 &= ~(1 << 23); /* unsupported request */
|
reg32 &= ~(1 << 23); /* unsupported request */
|
||||||
reg32 |= (1 << 31);
|
reg32 |= (1 << 31);
|
||||||
pci_write_config32(dev, 0x40, reg32);
|
pci_write_config32(dev, 0x40, reg32);
|
||||||
|
|
||||||
|
if (acpi_is_wakeup_s3()) {
|
||||||
|
/* Reset ports that are disabled or
|
||||||
|
* polling before returning to the OS. */
|
||||||
|
usb_xhci_reset_usb3(dev, 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void usb_xhci_set_subsystem(device_t dev, unsigned vendor,
|
static void usb_xhci_set_subsystem(device_t dev, unsigned vendor,
|
||||||
|
|
Loading…
Reference in a new issue