From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from mail02.groups.io (mail02.groups.io [66.175.222.108]) by spool.mail.gandi.net (Postfix) with ESMTPS id A9A77D80733 for ; Tue, 12 Sep 2023 18:07:04 +0000 (UTC) DKIM-Signature: a=rsa-sha256; bh=mWG8Naj2lmZaIB3cDHAmPAEaAhfbVCC47IyVZQldGqA=; c=relaxed/simple; d=groups.io; h=From:To:Cc:Subject:Date:Message-Id:MIME-Version:Precedence:List-Subscribe:List-Help:Sender:List-Id:Mailing-List:Delivered-To:Reply-To:List-Unsubscribe-Post:List-Unsubscribe:Content-Transfer-Encoding; s=20140610; t=1694542023; v=1; b=VAvFVX+I+3OSyQ4HdPrPz0tG/0p53ZJ6JZcLKX6zlmwMRNQvipZUki3ez9KfWVZ++KjaXTfO bdoqwzxQemj37LoCtFQqZflrY4UuZv1ngqOKGIL391npJ8lq5Z1ZlUX3JB7904APgIP6p6va19P TyUsK2gOQAXjDmGEatPDWGTY= X-Received: by 127.0.0.2 with SMTP id IaBoYY7687511x40oKd8Dy8P; Tue, 12 Sep 2023 11:07:03 -0700 X-Received: from mx0b-002e3701.pphosted.com (mx0b-002e3701.pphosted.com [148.163.143.35]) by mx.groups.io with SMTP id smtpd.web10.1740.1694542001882798165 for ; Tue, 12 Sep 2023 11:07:02 -0700 X-Received: from pps.filterd (m0134423.ppops.net [127.0.0.1]) by mx0b-002e3701.pphosted.com (8.17.1.19/8.17.1.19) with ESMTP id 38CHh2aO018440; Tue, 12 Sep 2023 18:06:31 GMT X-Received: from p1lg14881.it.hpe.com ([16.230.97.202]) by mx0b-002e3701.pphosted.com (PPS) with ESMTPS id 3t2vmgr6y1-1 (version=TLSv1.2 cipher=ECDHE-RSA-AES256-GCM-SHA384 bits=256 verify=NOT); Tue, 12 Sep 2023 18:06:30 +0000 X-Received: from p1lg14886.dc01.its.hpecorp.net (unknown [10.119.18.237]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature RSA-PSS (2048 bits) server-digest SHA256) (No client certificate requested) by p1lg14881.it.hpe.com (Postfix) with ESMTPS id 1DDB5805E5F; Tue, 12 Sep 2023 18:06:30 +0000 (UTC) X-Received: from cat.eag.rdlabs.hpecorp.net (unknown [16.231.227.39]) by p1lg14886.dc01.its.hpecorp.net (Postfix) with ESMTP id 9F74281260E; Tue, 12 Sep 2023 18:06:29 +0000 (UTC) X-Received: by cat.eag.rdlabs.hpecorp.net (Postfix, from userid 178665) id DEBDCAC763; Tue, 12 Sep 2023 13:06:28 -0500 (CDT) From: "Henz, Patrick" To: devel@edk2.groups.io Cc: hao.a.wu@intel.com, ray.ni@intel.com, Patrick Henz Subject: [edk2-devel] [PATCH v3] MdeModulePkg/XhciDxe: Use Performance Timer for XHCI Timeouts Date: Tue, 12 Sep 2023 13:05:57 -0500 Message-Id: <382ea3abf9d4e410cfcde1bcff453eac409e350b.1694541645.git.patrick.henz@hpe.com> X-Proofpoint-ORIG-GUID: -I6kDn5_cWKIGWMhTWYIj7mwFIyglLan X-Proofpoint-GUID: -I6kDn5_cWKIGWMhTWYIj7mwFIyglLan X-Proofpoint-UnRewURL: 0 URL was un-rewritten MIME-Version: 1.0 Precedence: Bulk List-Subscribe: List-Help: Sender: devel@edk2.groups.io List-Id: Mailing-List: list devel@edk2.groups.io; contact devel+owner@edk2.groups.io Reply-To: devel@edk2.groups.io,patrick.henz@hpe.com List-Unsubscribe-Post: List-Unsubscribe=One-Click List-Unsubscribe: X-Gm-Message-State: jGV1P8hITSZBUqMsf5BwWIgjx7686176AA= Content-Transfer-Encoding: quoted-printable X-GND-Status: LEGIT Authentication-Results: spool.mail.gandi.net; dkim=pass header.d=groups.io header.s=20140610 header.b=VAvFVX+I; spf=pass (spool.mail.gandi.net: domain of bounce@groups.io designates 66.175.222.108 as permitted sender) smtp.mailfrom=bounce@groups.io; dmarc=fail reason="SPF not aligned (relaxed), DKIM not aligned (relaxed)" header.from=hpe.com (policy=none) REF:https://bugzilla.tianocore.org/show_bug.cgi?id=3D2948 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 Cc: Ray Ni Signed-off-by: Patrick Henz Reviewed-by: --- MdeModulePkg/Bus/Pci/XhciDxe/Xhci.c | 117 +++++++++++++++++++++++ MdeModulePkg/Bus/Pci/XhciDxe/Xhci.h | 32 +++++++ MdeModulePkg/Bus/Pci/XhciDxe/XhciDxe.inf | 2 + MdeModulePkg/Bus/Pci/XhciDxe/XhciReg.c | 68 +++++-------- MdeModulePkg/Bus/Pci/XhciDxe/XhciSched.c | 72 ++++++-------- 5 files changed, 204 insertions(+), 87 deletions(-) diff --git a/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.c b/MdeModulePkg/Bus/Pci/Xhc= iDxe/Xhci.c index 62682dd27c..7a2e32a9dd 100644 --- a/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.c +++ b/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.c @@ -1,6 +1,7 @@ /** @file=0D The XHCI controller driver.=0D =0D +(C) Copyright 2023 Hewlett Packard Enterprise Development LP
=0D Copyright (c) 2011 - 2022, Intel Corporation. All rights reserved.
=0D SPDX-License-Identifier: BSD-2-Clause-Patent=0D =0D @@ -85,6 +86,11 @@ EFI_USB2_HC_PROTOCOL gXhciUsb2HcTemplate =3D { 0x0=0D };=0D =0D +UINT64 mPerformanceCounterStartValue;=0D +UINT64 mPerformanceCounterEndValue;=0D +UINT64 mPerformanceCounterFrequency;=0D +BOOLEAN mPerformanceCounterValuesCached =3D FALSE;=0D +=0D /**=0D Retrieves the capability of root hub ports.=0D =0D @@ -2294,3 +2300,114 @@ XhcDriverBindingStop ( =0D return EFI_SUCCESS;=0D }=0D +=0D +/**=0D + Converts a time in nanoseconds to a performance counter tick count.=0D +=0D + @param Time The time in nanoseconds to be converted to performance cou= nter ticks.=0D + @return Time in nanoseconds converted to ticks.=0D +**/=0D +UINT64=0D +XhcConvertTimeToTicks (=0D + IN UINT64 Time=0D + )=0D +{=0D + UINT64 Ticks;=0D + UINT64 Remainder;=0D + UINT64 Divisor;=0D + UINTN Shift;=0D +=0D + // Cache the return values to avoid repeated calls to GetPerformanceCoun= terProperties ()=0D + if (!mPerformanceCounterValuesCached) {=0D + mPerformanceCounterFrequency =3D GetPerformanceCounterProperties (=0D + &mPerformanceCounterStartValue,=0D + &mPerformanceCounterEndValue=0D + );=0D +=0D + mPerformanceCounterValuesCached =3D TRUE;=0D + }=0D +=0D + // Prevent returning a tick value of 0, unless Time is already 0=0D + if (0 =3D=3D mPerformanceCounterFrequency) {=0D + return Time;=0D + }=0D +=0D + // Nanoseconds per second=0D + Divisor =3D 1000000000;=0D +=0D + //=0D + // Frequency=0D + // Ticks =3D ------------- x Time=0D + // 1,000,000,000=0D + //=0D + Ticks =3D MultU64x64 (=0D + DivU64x64Remainder (=0D + mPerformanceCounterFrequency,=0D + Divisor,=0D + &Remainder=0D + ),=0D + Time=0D + );=0D +=0D + //=0D + // Ensure (Remainder * Time) will not overflow 64-bit.=0D + //=0D + // HighBitSet64 (Remainder) + 1 + HighBitSet64 (Time) + 1 <=3D 64=0D + //=0D + Shift =3D MAX (0, HighBitSet64 (Remainder) + HighBitSet64 (Time) - 6= 2);=0D + Remainder =3D RShiftU64 (Remainder, (UINTN)Shift);=0D + Divisor =3D RShiftU64 (Divisor, (UINTN)Shift);=0D + Ticks +=3D DivU64x64Remainder (MultU64x64 (Remainder, Time), Divisor,= NULL);=0D +=0D + return Ticks;=0D +}=0D +=0D +/**=0D + Computes and returns the elapsed ticks since PreviousTick. The=0D + value of PreviousTick is overwritten with the current performance=0D + counter value.=0D +=0D + @param PreviousTick Pointer to PreviousTick count.=0D + @return The elapsed ticks since PreviousCount. PreviousCount is=0D + overwritten with the current performance counter value.=0D +**/=0D +UINT64=0D +XhcGetElapsedTicks (=0D + IN OUT UINT64 *PreviousTick=0D + )=0D +{=0D + UINT64 CurrentTick;=0D + UINT64 Delta;=0D +=0D + CurrentTick =3D GetPerformanceCounter ();=0D +=0D + //=0D + // Determine if the counter is counting up or down=0D + //=0D + if (mPerformanceCounterStartValue < mPerformanceCounterEndValue) {=0D + //=0D + // Counter counts upwards, check for an overflow condition=0D + //=0D + if (*PreviousTick > CurrentTick) {=0D + Delta =3D (mPerformanceCounterEndValue - *PreviousTick) + CurrentTic= k;=0D + } else {=0D + Delta =3D CurrentTick - *PreviousTick;=0D + }=0D + } else {=0D + //=0D + // Counter counts downwards, check for an underflow condition=0D + //=0D + if (*PreviousTick < CurrentTick) {=0D + Delta =3D (mPerformanceCounterStartValue - CurrentTick) + *PreviousT= ick;=0D + } else {=0D + Delta =3D *PreviousTick - CurrentTick;=0D + }=0D + }=0D +=0D + //=0D + // Set PreviousTick to CurrentTick=0D + //=0D + *PreviousTick =3D CurrentTick;=0D +=0D + return Delta;=0D +}=0D diff --git a/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.h b/MdeModulePkg/Bus/Pci/Xhc= iDxe/Xhci.h index ca223bd20c..4401675872 100644 --- a/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.h +++ b/MdeModulePkg/Bus/Pci/XhciDxe/Xhci.h @@ -2,6 +2,7 @@ =0D Provides some data structure definitions used by the XHCI host controlle= r driver.=0D =0D +(C) Copyright 2023 Hewlett Packard Enterprise Development LP
=0D Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.
=0D Copyright (c) Microsoft Corporation.
=0D SPDX-License-Identifier: BSD-2-Clause-Patent=0D @@ -26,6 +27,7 @@ SPDX-License-Identifier: BSD-2-Clause-Patent #include =0D #include =0D #include =0D +#include =0D =0D #include =0D =0D @@ -37,6 +39,11 @@ typedef struct _USB_DEV_CONTEXT USB_DEV_CONTEXT; #include "ComponentName.h"=0D #include "UsbHcMem.h"=0D =0D +//=0D +// Converts a count from microseconds to nanoseconds=0D +//=0D +#define XHC_MICROSECOND_TO_NANOSECOND(Time) (MultU64x32((Time), 1000))=0D +=0D //=0D // The unit is microsecond, setting it as 1us.=0D //=0D @@ -720,4 +727,29 @@ XhcAsyncIsochronousTransfer ( IN VOID *Context=0D );=0D =0D +/**=0D + Converts a time in nanoseconds to a performance counter tick count.=0D +=0D + @param Time The time in nanoseconds to be converted to performance cou= nter ticks.=0D + @return Time in nanoseconds converted to ticks.=0D +**/=0D +UINT64=0D +XhcConvertTimeToTicks (=0D + UINT64 Time=0D + );=0D +=0D +/**=0D + Computes and returns the elapsed ticks since PreviousTick. The=0D + value of PreviousTick is overwritten with the current performance=0D + counter value.=0D +=0D + @param PreviousTick Pointer to PreviousTick count.=0D + @return The elapsed ticks since PreviousCount. PreviousCount is=0D + overwritten with the current performance counter value.=0D +**/=0D +UINT64=0D +XhcGetElapsedTicks (=0D + IN OUT UINT64 *PreviousTick=0D + );=0D +=0D #endif=0D diff --git a/MdeModulePkg/Bus/Pci/XhciDxe/XhciDxe.inf b/MdeModulePkg/Bus/Pc= i/XhciDxe/XhciDxe.inf index 5865d86822..18ef87916a 100644 --- a/MdeModulePkg/Bus/Pci/XhciDxe/XhciDxe.inf +++ b/MdeModulePkg/Bus/Pci/XhciDxe/XhciDxe.inf @@ -3,6 +3,7 @@ # It implements the interfaces of monitoring the status of all ports and = transferring=0D # Control, Bulk, Interrupt and Isochronous requests to those attached usb= LS/FS/HS/SS devices.=0D #=0D +# (C) Copyright 2023 Hewlett Packard Enterprise Development LP
=0D # Copyright (c) 2011 - 2018, Intel Corporation. All rights reserved.
= =0D #=0D # SPDX-License-Identifier: BSD-2-Clause-Patent=0D @@ -54,6 +55,7 @@ BaseMemoryLib=0D DebugLib=0D ReportStatusCodeLib=0D + TimerLib=0D =0D [Guids]=0D gEfiEventExitBootServicesGuid ## SOMETIMES_CONSUMES ## E= vent=0D diff --git a/MdeModulePkg/Bus/Pci/XhciDxe/XhciReg.c b/MdeModulePkg/Bus/Pci/= XhciDxe/XhciReg.c index 5700fc5fb8..40f2f1f227 100644 --- a/MdeModulePkg/Bus/Pci/XhciDxe/XhciReg.c +++ b/MdeModulePkg/Bus/Pci/XhciDxe/XhciReg.c @@ -2,6 +2,7 @@ =0D The XHCI register operation routines.=0D =0D +(C) Copyright 2023 Hewlett Packard Enterprise Development LP
=0D Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.
=0D SPDX-License-Identifier: BSD-2-Clause-Patent=0D =0D @@ -417,15 +418,14 @@ XhcClearOpRegBit ( Wait the operation register's bit as specified by Bit=0D to become set (or clear).=0D =0D - @param Xhc The XHCI Instance.=0D - @param Offset The offset of the operation register.=0D - @param Bit The bit of the register to wait for.=0D - @param WaitToSet Wait the bit to set or clear.=0D - @param Timeout The time to wait before abort (in millise= cond, ms).=0D + @param Xhc The XHCI Instance.=0D + @param Offset The offset of the operation register.=0D + @param Bit The bit of the register to wait for.=0D + @param WaitToSet Wait the bit to set or clear.=0D + @param Timeout The time to wait before abort (in millisecond, ms).= =0D =0D - @retval EFI_SUCCESS The bit successfully changed by host cont= roller.=0D - @retval EFI_TIMEOUT The time out occurred.=0D - @retval EFI_OUT_OF_RESOURCES Memory for the timer event could not be a= llocated.=0D + @retval EFI_SUCCESS The bit successfully changed by host controller.=0D + @retval EFI_TIMEOUT The time out occurred.=0D =0D **/=0D EFI_STATUS=0D @@ -437,54 +437,34 @@ XhcWaitOpRegBit ( IN UINT32 Timeout=0D )=0D {=0D - EFI_STATUS Status;=0D - EFI_EVENT TimeoutEvent;=0D -=0D - TimeoutEvent =3D NULL;=0D + UINT64 TimeoutTicks;=0D + UINT64 ElapsedTicks;=0D + UINT64 TicksDelta;=0D + UINT64 CurrentTick;=0D =0D if (Timeout =3D=3D 0) {=0D return EFI_TIMEOUT;=0D }=0D =0D - Status =3D gBS->CreateEvent (=0D - EVT_TIMER,=0D - TPL_CALLBACK,=0D - NULL,=0D - NULL,=0D - &TimeoutEvent=0D - );=0D -=0D - if (EFI_ERROR (Status)) {=0D - goto DONE;=0D - }=0D -=0D - Status =3D gBS->SetTimer (=0D - TimeoutEvent,=0D - TimerRelative,=0D - EFI_TIMER_PERIOD_MILLISECONDS (Timeout)=0D - );=0D -=0D - if (EFI_ERROR (Status)) {=0D - goto DONE;=0D - }=0D -=0D + TimeoutTicks =3D XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND (T= imeout * XHC_1_MILLISECOND));=0D + ElapsedTicks =3D 0;=0D + CurrentTick =3D GetPerformanceCounter ();=0D do {=0D if (XHC_REG_BIT_IS_SET (Xhc, Offset, Bit) =3D=3D WaitToSet) {=0D - Status =3D EFI_SUCCESS;=0D - goto DONE;=0D + return EFI_SUCCESS;=0D }=0D =0D gBS->Stall (XHC_1_MICROSECOND);=0D - } while (EFI_ERROR (gBS->CheckEvent (TimeoutEvent)));=0D -=0D - Status =3D EFI_TIMEOUT;=0D + TicksDelta =3D XhcGetElapsedTicks (&CurrentTick);=0D + // Ensure that ElapsedTicks is always incremented to avoid indefinite = hangs=0D + if (TicksDelta =3D=3D 0) {=0D + TicksDelta =3D XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND = (XHC_1_MICROSECOND));=0D + }=0D =0D -DONE:=0D - if (TimeoutEvent !=3D NULL) {=0D - gBS->CloseEvent (TimeoutEvent);=0D - }=0D + ElapsedTicks +=3D TicksDelta;=0D + } while (ElapsedTicks < TimeoutTicks);=0D =0D - return Status;=0D + return EFI_TIMEOUT;=0D }=0D =0D /**=0D diff --git a/MdeModulePkg/Bus/Pci/XhciDxe/XhciSched.c b/MdeModulePkg/Bus/Pc= i/XhciDxe/XhciSched.c index 53421e64a8..613b1485f1 100644 --- a/MdeModulePkg/Bus/Pci/XhciDxe/XhciSched.c +++ b/MdeModulePkg/Bus/Pci/XhciDxe/XhciSched.c @@ -2,6 +2,7 @@ =0D XHCI transfer scheduling routines.=0D =0D +(C) Copyright 2023 Hewlett Packard Enterprise Development LP
=0D Copyright (c) 2011 - 2020, Intel Corporation. All rights reserved.
=0D Copyright (c) Microsoft Corporation.
=0D Copyright (C) 2022 Advanced Micro Devices, Inc. All rights reserved.
=0D @@ -1276,15 +1277,14 @@ EXIT: /**=0D Execute the transfer by polling the URB. This is a synchronous operation= .=0D =0D - @param Xhc The XHCI Instance.=0D - @param CmdTransfer The executed URB is for cmd transfer or n= ot.=0D - @param Urb The URB to execute.=0D - @param Timeout The time to wait before abort, in millise= cond.=0D + @param Xhc The XHCI Instance.=0D + @param CmdTransfer The executed URB is for cmd transfer or not.=0D + @param Urb The URB to execute.=0D + @param Timeout The time to wait before abort, in millisecond.= =0D =0D - @return EFI_DEVICE_ERROR The transfer failed due to transfer error= .=0D - @return EFI_TIMEOUT The transfer failed due to time out.=0D - @return EFI_SUCCESS The transfer finished OK.=0D - @retval EFI_OUT_OF_RESOURCES Memory for the timer event could not be a= llocated.=0D + @return EFI_DEVICE_ERROR The transfer failed due to transfer error.=0D + @return EFI_TIMEOUT The transfer failed due to time out.=0D + @return EFI_SUCCESS The transfer finished OK.=0D =0D **/=0D EFI_STATUS=0D @@ -1299,12 +1299,14 @@ XhcExecTransfer ( UINT8 SlotId;=0D UINT8 Dci;=0D BOOLEAN Finished;=0D - EFI_EVENT TimeoutEvent;=0D + UINT64 TimeoutTicks;=0D + UINT64 ElapsedTicks;=0D + UINT64 TicksDelta;=0D + UINT64 CurrentTick;=0D BOOLEAN IndefiniteTimeout;=0D =0D Status =3D EFI_SUCCESS;=0D Finished =3D FALSE;=0D - TimeoutEvent =3D NULL;=0D IndefiniteTimeout =3D FALSE;=0D =0D if (CmdTransfer) {=0D @@ -1322,34 +1324,18 @@ XhcExecTransfer ( =0D if (Timeout =3D=3D 0) {=0D IndefiniteTimeout =3D TRUE;=0D - goto RINGDOORBELL;=0D - }=0D -=0D - Status =3D gBS->CreateEvent (=0D - EVT_TIMER,=0D - TPL_CALLBACK,=0D - NULL,=0D - NULL,=0D - &TimeoutEvent=0D - );=0D -=0D - if (EFI_ERROR (Status)) {=0D - goto DONE;=0D - }=0D -=0D - Status =3D gBS->SetTimer (=0D - TimeoutEvent,=0D - TimerRelative,=0D - EFI_TIMER_PERIOD_MILLISECONDS (Timeout)=0D - );=0D -=0D - if (EFI_ERROR (Status)) {=0D - goto DONE;=0D }=0D =0D -RINGDOORBELL:=0D XhcRingDoorBell (Xhc, SlotId, Dci);=0D =0D + TimeoutTicks =3D XhcConvertTimeToTicks (=0D + XHC_MICROSECOND_TO_NANOSECOND (=0D + Timeout * XHC_1_MILLISECOND=0D + )=0D + );=0D + ElapsedTicks =3D 0;=0D + CurrentTick =3D GetPerformanceCounter ();=0D +=0D do {=0D Finished =3D XhcCheckUrbResult (Xhc, Urb);=0D if (Finished) {=0D @@ -1357,22 +1343,22 @@ RINGDOORBELL: }=0D =0D gBS->Stall (XHC_1_MICROSECOND);=0D - } while (IndefiniteTimeout || EFI_ERROR (gBS->CheckEvent (TimeoutEvent))= );=0D + TicksDelta =3D XhcGetElapsedTicks (&CurrentTick);=0D + // Ensure that ElapsedTicks is always incremented to avoid indefinite = hangs=0D + if (TicksDelta =3D=3D 0) {=0D + TicksDelta =3D XhcConvertTimeToTicks (XHC_MICROSECOND_TO_NANOSECOND = (XHC_1_MICROSECOND));=0D + }=0D =0D -DONE:=0D - if (EFI_ERROR (Status)) {=0D - Urb->Result =3D EFI_USB_ERR_NOTEXECUTE;=0D - } else if (!Finished) {=0D + ElapsedTicks +=3D TicksDelta;=0D + } while (IndefiniteTimeout || ElapsedTicks < TimeoutTicks);=0D +=0D + if (!Finished) {=0D Urb->Result =3D EFI_USB_ERR_TIMEOUT;=0D Status =3D EFI_TIMEOUT;=0D } else if (Urb->Result !=3D EFI_USB_NOERROR) {=0D Status =3D EFI_DEVICE_ERROR;=0D }=0D =0D - if (TimeoutEvent !=3D NULL) {=0D - gBS->CloseEvent (TimeoutEvent);=0D - }=0D -=0D return Status;=0D }=0D =0D --=20 2.34.1 -=-=-=-=-=-=-=-=-=-=-=- Groups.io Links: You receive all messages sent to this group. View/Reply Online (#108549): https://edk2.groups.io/g/devel/message/108549 Mute This Topic: https://groups.io/mt/101320913/7686176 Group Owner: devel+owner@edk2.groups.io Unsubscribe: https://edk2.groups.io/g/devel/unsub [rebecca@openfw.io] -=-=-=-=-=-=-=-=-=-=-=-