MdeModulePkg/XhciDxe: Use Performance Timer for XHCI Timeouts

REF:https://bugzilla.tianocore.org/show_bug.cgi?id=2948

XhciDxe uses the timer functionality provided by the
boot services table to detect timeout conditions. This
breaks the driver's ExitBootServices call back, as
CoreExitBootServices halts the timer before signaling
the ExitBootServices event. If the host controller
fails to halt in the call back, the timeout condition
will never occur and the boot gets stuck in an indefinite
spin loop. Use the free running timer provided by
TimerLib to calculate timeouts, avoiding the potential
hang.

Cc: Hao A Wu <hao.a.wu@intel.com>
Cc: Ray Ni <ray.ni@intel.com>
Signed-off-by: Patrick Henz <patrick.henz@hpe.com>
Reviewed-by: Hao A Wu <hao.a.wu@intel.com>
This commit is contained in:
Henz, Patrick 2023-09-13 02:05:57 +08:00 committed by mergify[bot]
parent 8b2e6b90b8
commit 43dcf453fc
5 changed files with 204 additions and 87 deletions

View File

@ -1,6 +1,7 @@
/** @file /** @file
The XHCI controller driver. The XHCI controller driver.
(C) Copyright 2023 Hewlett Packard Enterprise Development LP<BR>
Copyright (c) 2011 - 2022, Intel Corporation. All rights reserved.<BR> Copyright (c) 2011 - 2022, Intel Corporation. All rights reserved.<BR>
SPDX-License-Identifier: BSD-2-Clause-Patent SPDX-License-Identifier: BSD-2-Clause-Patent
@ -85,6 +86,11 @@ EFI_USB2_HC_PROTOCOL gXhciUsb2HcTemplate = {
0x0 0x0
}; };
UINT64 mPerformanceCounterStartValue;
UINT64 mPerformanceCounterEndValue;
UINT64 mPerformanceCounterFrequency;
BOOLEAN mPerformanceCounterValuesCached = FALSE;
/** /**
Retrieves the capability of root hub ports. Retrieves the capability of root hub ports.
@ -2294,3 +2300,114 @@ XhcDriverBindingStop (
return EFI_SUCCESS; return EFI_SUCCESS;
} }
/**
Converts a time in nanoseconds to a performance counter tick count.
@param Time The time in nanoseconds to be converted to performance counter ticks.
@return Time in nanoseconds converted to ticks.
**/
UINT64
XhcConvertTimeToTicks (
IN UINT64 Time
)
{
UINT64 Ticks;
UINT64 Remainder;
UINT64 Divisor;
UINTN Shift;
// Cache the return values to avoid repeated calls to GetPerformanceCounterProperties ()
if (!mPerformanceCounterValuesCached) {
mPerformanceCounterFrequency = GetPerformanceCounterProperties (
&mPerformanceCounterStartValue,
&mPerformanceCounterEndValue
);
mPerformanceCounterValuesCached = TRUE;
}
// Prevent returning a tick value of 0, unless Time is already 0
if (0 == mPerformanceCounterFrequency) {
return Time;
}
// Nanoseconds per second
Divisor = 1000000000;
//
// Frequency
// Ticks = ------------- x Time
// 1,000,000,000
//
Ticks = MultU64x64 (
DivU64x64Remainder (
mPerformanceCounterFrequency,
Divisor,
&Remainder
),
Time
);
//
// Ensure (Remainder * Time) will not overflow 64-bit.
//
// HighBitSet64 (Remainder) + 1 + HighBitSet64 (Time) + 1 <= 64
//
Shift = MAX (0, HighBitSet64 (Remainder) + HighBitSet64 (Time) - 62);
Remainder = RShiftU64 (Remainder, (UINTN)Shift);
Divisor = RShiftU64 (Divisor, (UINTN)Shift);
Ticks += DivU64x64Remainder (MultU64x64 (Remainder, Time), Divisor, NULL);
return Ticks;
}
/**
Computes and returns the elapsed ticks since PreviousTick. The
value of PreviousTick is overwritten with the current performance
counter value.
@param PreviousTick Pointer to PreviousTick count.
@return The elapsed ticks since PreviousCount. PreviousCount is
overwritten with the current performance counter value.
**/
UINT64
XhcGetElapsedTicks (
IN OUT UINT64 *PreviousTick
)
{
UINT64 CurrentTick;
UINT64 Delta;
CurrentTick = GetPerformanceCounter ();
//
// Determine if the counter is counting up or down
//
if (mPerformanceCounterStartValue < mPerformanceCounterEndValue) {
//
// Counter counts upwards, check for an overflow condition
//
if (*PreviousTick > CurrentTick) {
Delta = (mPerformanceCounterEndValue - *PreviousTick) + CurrentTick;
} else {
Delta = CurrentTick - *PreviousTick;
}
} else {
//
// Counter counts downwards, check for an underflow condition
//
if (*PreviousTick < CurrentTick) {
Delta = (mPerformanceCounterStartValue - CurrentTick) + *PreviousTick;
} else {
Delta = *PreviousTick - CurrentTick;
}
}
//
// Set PreviousTick to CurrentTick
//
*PreviousTick = CurrentTick;
return Delta;
}

View File

@ -2,6 +2,7 @@
Provides some data structure definitions used by the XHCI host controller driver. Provides some data structure definitions used by the XHCI host controller driver.
(C) Copyright 2023 Hewlett Packard Enterprise Development LP<BR>
Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.<BR> Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.<BR>
Copyright (c) Microsoft Corporation.<BR> Copyright (c) Microsoft Corporation.<BR>
SPDX-License-Identifier: BSD-2-Clause-Patent SPDX-License-Identifier: BSD-2-Clause-Patent
@ -26,6 +27,7 @@ SPDX-License-Identifier: BSD-2-Clause-Patent
#include <Library/UefiLib.h> #include <Library/UefiLib.h>
#include <Library/DebugLib.h> #include <Library/DebugLib.h>
#include <Library/ReportStatusCodeLib.h> #include <Library/ReportStatusCodeLib.h>
#include <Library/TimerLib.h>
#include <IndustryStandard/Pci.h> #include <IndustryStandard/Pci.h>
@ -37,6 +39,11 @@ typedef struct _USB_DEV_CONTEXT USB_DEV_CONTEXT;
#include "ComponentName.h" #include "ComponentName.h"
#include "UsbHcMem.h" #include "UsbHcMem.h"
//
// Converts a count from microseconds to nanoseconds
//
#define XHC_MICROSECOND_TO_NANOSECOND(Time) (MultU64x32((Time), 1000))
// //
// The unit is microsecond, setting it as 1us. // The unit is microsecond, setting it as 1us.
// //
@ -720,4 +727,29 @@ XhcAsyncIsochronousTransfer (
IN VOID *Context IN VOID *Context
); );
/**
Converts a time in nanoseconds to a performance counter tick count.
@param Time The time in nanoseconds to be converted to performance counter ticks.
@return Time in nanoseconds converted to ticks.
**/
UINT64
XhcConvertTimeToTicks (
UINT64 Time
);
/**
Computes and returns the elapsed ticks since PreviousTick. The
value of PreviousTick is overwritten with the current performance
counter value.
@param PreviousTick Pointer to PreviousTick count.
@return The elapsed ticks since PreviousCount. PreviousCount is
overwritten with the current performance counter value.
**/
UINT64
XhcGetElapsedTicks (
IN OUT UINT64 *PreviousTick
);
#endif #endif

View File

@ -3,6 +3,7 @@
# It implements the interfaces of monitoring the status of all ports and transferring # It implements the interfaces of monitoring the status of all ports and transferring
# Control, Bulk, Interrupt and Isochronous requests to those attached usb LS/FS/HS/SS devices. # Control, Bulk, Interrupt and Isochronous requests to those attached usb LS/FS/HS/SS devices.
# #
# (C) Copyright 2023 Hewlett Packard Enterprise Development LP<BR>
# Copyright (c) 2011 - 2018, Intel Corporation. All rights reserved.<BR> # Copyright (c) 2011 - 2018, Intel Corporation. All rights reserved.<BR>
# #
# SPDX-License-Identifier: BSD-2-Clause-Patent # SPDX-License-Identifier: BSD-2-Clause-Patent
@ -54,6 +55,7 @@
BaseMemoryLib BaseMemoryLib
DebugLib DebugLib
ReportStatusCodeLib ReportStatusCodeLib
TimerLib
[Guids] [Guids]
gEfiEventExitBootServicesGuid ## SOMETIMES_CONSUMES ## Event gEfiEventExitBootServicesGuid ## SOMETIMES_CONSUMES ## Event

View File

@ -2,6 +2,7 @@
The XHCI register operation routines. The XHCI register operation routines.
(C) Copyright 2023 Hewlett Packard Enterprise Development LP<BR>
Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.<BR> Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.<BR>
SPDX-License-Identifier: BSD-2-Clause-Patent SPDX-License-Identifier: BSD-2-Clause-Patent
@ -417,15 +418,14 @@ XhcClearOpRegBit (
Wait the operation register's bit as specified by Bit Wait the operation register's bit as specified by Bit
to become set (or clear). to become set (or clear).
@param Xhc The XHCI Instance. @param Xhc The XHCI Instance.
@param Offset The offset of the operation register. @param Offset The offset of the operation register.
@param Bit The bit of the register to wait for. @param Bit The bit of the register to wait for.
@param WaitToSet Wait the bit to set or clear. @param WaitToSet Wait the bit to set or clear.
@param Timeout The time to wait before abort (in millisecond, ms). @param Timeout The time to wait before abort (in millisecond, ms).
@retval EFI_SUCCESS The bit successfully changed by host controller. @retval EFI_SUCCESS The bit successfully changed by host controller.
@retval EFI_TIMEOUT The time out occurred. @retval EFI_TIMEOUT The time out occurred.
@retval EFI_OUT_OF_RESOURCES Memory for the timer event could not be allocated.
**/ **/
EFI_STATUS EFI_STATUS
@ -437,54 +437,34 @@ XhcWaitOpRegBit (
IN UINT32 Timeout IN UINT32 Timeout
) )
{ {
EFI_STATUS Status; UINT64 TimeoutTicks;
EFI_EVENT TimeoutEvent; UINT64 ElapsedTicks;
UINT64 TicksDelta;
TimeoutEvent = NULL; UINT64 CurrentTick;
if (Timeout == 0) { if (Timeout == 0) {
return EFI_TIMEOUT; return EFI_TIMEOUT;
} }
Status = gBS->CreateEvent ( TimeoutTicks = XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND (Timeout * XHC_1_MILLISECOND));
EVT_TIMER, ElapsedTicks = 0;
TPL_CALLBACK, CurrentTick = GetPerformanceCounter ();
NULL,
NULL,
&TimeoutEvent
);
if (EFI_ERROR (Status)) {
goto DONE;
}
Status = gBS->SetTimer (
TimeoutEvent,
TimerRelative,
EFI_TIMER_PERIOD_MILLISECONDS (Timeout)
);
if (EFI_ERROR (Status)) {
goto DONE;
}
do { do {
if (XHC_REG_BIT_IS_SET (Xhc, Offset, Bit) == WaitToSet) { if (XHC_REG_BIT_IS_SET (Xhc, Offset, Bit) == WaitToSet) {
Status = EFI_SUCCESS; return EFI_SUCCESS;
goto DONE;
} }
gBS->Stall (XHC_1_MICROSECOND); gBS->Stall (XHC_1_MICROSECOND);
} while (EFI_ERROR (gBS->CheckEvent (TimeoutEvent))); TicksDelta = XhcGetElapsedTicks (&CurrentTick);
// Ensure that ElapsedTicks is always incremented to avoid indefinite hangs
if (TicksDelta == 0) {
TicksDelta = XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND (XHC_1_MICROSECOND));
}
Status = EFI_TIMEOUT; ElapsedTicks += TicksDelta;
} while (ElapsedTicks < TimeoutTicks);
DONE: return EFI_TIMEOUT;
if (TimeoutEvent != NULL) {
gBS->CloseEvent (TimeoutEvent);
}
return Status;
} }
/** /**

View File

@ -2,6 +2,7 @@
XHCI transfer scheduling routines. XHCI transfer scheduling routines.
(C) Copyright 2023 Hewlett Packard Enterprise Development LP<BR>
Copyright (c) 2011 - 2020, Intel Corporation. All rights reserved.<BR> Copyright (c) 2011 - 2020, Intel Corporation. All rights reserved.<BR>
Copyright (c) Microsoft Corporation.<BR> Copyright (c) Microsoft Corporation.<BR>
Copyright (C) 2022 Advanced Micro Devices, Inc. All rights reserved.<BR> Copyright (C) 2022 Advanced Micro Devices, Inc. All rights reserved.<BR>
@ -1276,15 +1277,14 @@ EXIT:
/** /**
Execute the transfer by polling the URB. This is a synchronous operation. Execute the transfer by polling the URB. This is a synchronous operation.
@param Xhc The XHCI Instance. @param Xhc The XHCI Instance.
@param CmdTransfer The executed URB is for cmd transfer or not. @param CmdTransfer The executed URB is for cmd transfer or not.
@param Urb The URB to execute. @param Urb The URB to execute.
@param Timeout The time to wait before abort, in millisecond. @param Timeout The time to wait before abort, in millisecond.
@return EFI_DEVICE_ERROR The transfer failed due to transfer error. @return EFI_DEVICE_ERROR The transfer failed due to transfer error.
@return EFI_TIMEOUT The transfer failed due to time out. @return EFI_TIMEOUT The transfer failed due to time out.
@return EFI_SUCCESS The transfer finished OK. @return EFI_SUCCESS The transfer finished OK.
@retval EFI_OUT_OF_RESOURCES Memory for the timer event could not be allocated.
**/ **/
EFI_STATUS EFI_STATUS
@ -1299,12 +1299,14 @@ XhcExecTransfer (
UINT8 SlotId; UINT8 SlotId;
UINT8 Dci; UINT8 Dci;
BOOLEAN Finished; BOOLEAN Finished;
EFI_EVENT TimeoutEvent; UINT64 TimeoutTicks;
UINT64 ElapsedTicks;
UINT64 TicksDelta;
UINT64 CurrentTick;
BOOLEAN IndefiniteTimeout; BOOLEAN IndefiniteTimeout;
Status = EFI_SUCCESS; Status = EFI_SUCCESS;
Finished = FALSE; Finished = FALSE;
TimeoutEvent = NULL;
IndefiniteTimeout = FALSE; IndefiniteTimeout = FALSE;
if (CmdTransfer) { if (CmdTransfer) {
@ -1322,34 +1324,18 @@ XhcExecTransfer (
if (Timeout == 0) { if (Timeout == 0) {
IndefiniteTimeout = TRUE; IndefiniteTimeout = TRUE;
goto RINGDOORBELL;
} }
Status = gBS->CreateEvent (
EVT_TIMER,
TPL_CALLBACK,
NULL,
NULL,
&TimeoutEvent
);
if (EFI_ERROR (Status)) {
goto DONE;
}
Status = gBS->SetTimer (
TimeoutEvent,
TimerRelative,
EFI_TIMER_PERIOD_MILLISECONDS (Timeout)
);
if (EFI_ERROR (Status)) {
goto DONE;
}
RINGDOORBELL:
XhcRingDoorBell (Xhc, SlotId, Dci); XhcRingDoorBell (Xhc, SlotId, Dci);
TimeoutTicks = XhcConvertTimeToTicks (
XHC_MICROSECOND_TO_NANOSECOND (
Timeout * XHC_1_MILLISECOND
)
);
ElapsedTicks = 0;
CurrentTick = GetPerformanceCounter ();
do { do {
Finished = XhcCheckUrbResult (Xhc, Urb); Finished = XhcCheckUrbResult (Xhc, Urb);
if (Finished) { if (Finished) {
@ -1357,22 +1343,22 @@ RINGDOORBELL:
} }
gBS->Stall (XHC_1_MICROSECOND); gBS->Stall (XHC_1_MICROSECOND);
} while (IndefiniteTimeout || EFI_ERROR (gBS->CheckEvent (TimeoutEvent))); TicksDelta = XhcGetElapsedTicks (&CurrentTick);
// Ensure that ElapsedTicks is always incremented to avoid indefinite hangs
if (TicksDelta == 0) {
TicksDelta = XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND (XHC_1_MICROSECOND));
}
DONE: ElapsedTicks += TicksDelta;
if (EFI_ERROR (Status)) { } while (IndefiniteTimeout || ElapsedTicks < TimeoutTicks);
Urb->Result = EFI_USB_ERR_NOTEXECUTE;
} else if (!Finished) { if (!Finished) {
Urb->Result = EFI_USB_ERR_TIMEOUT; Urb->Result = EFI_USB_ERR_TIMEOUT;
Status = EFI_TIMEOUT; Status = EFI_TIMEOUT;
} else if (Urb->Result != EFI_USB_NOERROR) { } else if (Urb->Result != EFI_USB_NOERROR) {
Status = EFI_DEVICE_ERROR; Status = EFI_DEVICE_ERROR;
} }
if (TimeoutEvent != NULL) {
gBS->CloseEvent (TimeoutEvent);
}
return Status; return Status;
} }